Commit 3742c04d authored by Michael Hunold's avatar Michael Hunold Committed by Linus Torvalds

[PATCH] DVB: frontend conversion #4

- [DVB] nxt6000, sp887x: convert from dvb-i2c to kernel-i2c, MODULE_PARM()
  to module_param(), dvb_delay() to mdelay()

- [DVB] sp887x: move from home-brewn firmware loading to firmware_class
Signed-off-by: default avatarMichael Hunold <hunold@linuxtv.org>
Signed-off-by: default avatarAndrew Morton <akpm@osdl.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@osdl.org>
parent 67dfafeb
...@@ -36,11 +36,11 @@ ...@@ -36,11 +36,11 @@
#include "dvb_frontend.h" #include "dvb_frontend.h"
#include "nxt6000.h" #include "nxt6000.h"
static int debug = 0;
MODULE_DESCRIPTION("NxtWave NXT6000 DVB demodulator driver"); MODULE_DESCRIPTION("NxtWave NXT6000 DVB demodulator driver");
MODULE_AUTHOR("Florian Schirmer"); MODULE_AUTHOR("Florian Schirmer");
MODULE_LICENSE("GPL"); MODULE_LICENSE("GPL");
static int debug = 0;
MODULE_PARM(debug, "i"); MODULE_PARM(debug, "i");
static struct dvb_frontend_info nxt6000_info = { static struct dvb_frontend_info nxt6000_info = {
...@@ -68,43 +68,38 @@ struct nxt6000_config { ...@@ -68,43 +68,38 @@ struct nxt6000_config {
u8 tuner_addr; u8 tuner_addr;
u8 tuner_type; u8 tuner_type;
u8 clock_inversion; u8 clock_inversion;
struct i2c_adapter *i2c;
struct dvb_adapter *dvb;
}; };
#define TUNER_TYPE_ALP510 0 #define TUNER_TYPE_ALP510 0
#define TUNER_TYPE_SP5659 1 #define TUNER_TYPE_SP5659 1
#define TUNER_TYPE_SP5730 2 #define TUNER_TYPE_SP5730 2
#define FE2NXT(fe) ((struct nxt6000_config *)((fe)->data)) // #define FE2NXT(fe) ((struct nxt6000_config *)((fe)->data))
#define FREQ2DIV(freq) ((freq + 36166667) / 166667) #define FREQ2DIV(freq) ((freq + 36166667) / 166667)
#define dprintk if (debug) printk #define dprintk if (debug) printk
static int nxt6000_write(struct dvb_i2c_bus *i2c, u8 addr, u8 reg, u8 data) static int nxt6000_write(struct i2c_adapter *i2c, u8 addr, u8 reg, u8 data)
{ {
u8 buf[] = {reg, data}; u8 buf[] = {reg, data};
struct i2c_msg msg = {.addr = addr >> 1, .flags = 0, .buf = buf, .len = 2}; struct i2c_msg msg = {.addr = addr >> 1, .flags = 0, .buf = buf, .len = 2};
int ret; int ret;
if ((ret = i2c->xfer(i2c, &msg, 1)) != 1) if ((ret = i2c_transfer(i2c, &msg, 1)) != 1)
dprintk("nxt6000: nxt6000_write error (.addr = 0x%02X, reg: 0x%02X, data: 0x%02X, ret: %d)\n", addr, reg, data, ret); dprintk("nxt6000: nxt6000_write error (.addr = 0x%02X, reg: 0x%02X, data: 0x%02X, ret: %d)\n", addr, reg, data, ret);
return (ret != 1) ? -EFAULT : 0; return (ret != 1) ? -EFAULT : 0;
} }
static u8 nxt6000_writereg(struct dvb_frontend *fe, u8 reg, u8 data) static u8 nxt6000_writereg(struct nxt6000_config *nxt, u8 reg, u8 data)
{ {
return nxt6000_write(nxt->i2c, nxt->demod_addr, reg, data);
struct nxt6000_config *nxt = FE2NXT(fe);
return nxt6000_write(fe->i2c, nxt->demod_addr, reg, data);
} }
static u8 nxt6000_read(struct dvb_i2c_bus *i2c, u8 addr, u8 reg) static u8 nxt6000_read(struct i2c_adapter *i2c, u8 addr, u8 reg)
{ {
int ret; int ret;
u8 b0[] = {reg}; u8 b0[] = {reg};
u8 b1[] = {0}; u8 b1[] = {0};
...@@ -113,58 +108,50 @@ static u8 nxt6000_read(struct dvb_i2c_bus *i2c, u8 addr, u8 reg) ...@@ -113,58 +108,50 @@ static u8 nxt6000_read(struct dvb_i2c_bus *i2c, u8 addr, u8 reg)
{.addr = addr >> 1,.flags = I2C_M_RD,.buf = b1,.len = 1} {.addr = addr >> 1,.flags = I2C_M_RD,.buf = b1,.len = 1}
}; };
ret = i2c->xfer(i2c, msgs, 2); ret = i2c_transfer(i2c, msgs, 2);
if (ret != 2) if (ret != 2)
dprintk("nxt6000: nxt6000_read error (.addr = 0x%02X, reg: 0x%02X, ret: %d)\n", addr, reg, ret); dprintk("nxt6000: nxt6000_read error (.addr = 0x%02X, reg: 0x%02X, ret: %d)\n", addr, reg, ret);
return b1[0]; return b1[0];
} }
static u8 nxt6000_readreg(struct dvb_frontend *fe, u8 reg) static u8 nxt6000_readreg(struct nxt6000_config *nxt, u8 reg)
{ {
return nxt6000_read(nxt->i2c, nxt->demod_addr, reg);
struct nxt6000_config *nxt = FE2NXT(fe);
return nxt6000_read(fe->i2c, nxt->demod_addr, reg);
} }
static int pll_test(struct dvb_i2c_bus *i2c, u8 demod_addr, u8 tuner_addr) static int pll_test(struct i2c_adapter *i2c, u8 demod_addr, u8 tuner_addr)
{ {
u8 buf [1]; u8 buf [1];
struct i2c_msg msg = {.addr = tuner_addr >> 1,.flags = I2C_M_RD,.buf = buf,.len = 1 }; struct i2c_msg msg = {.addr = tuner_addr >> 1,.flags = I2C_M_RD,.buf = buf,.len = 1 };
int ret; int ret;
nxt6000_write(i2c, demod_addr, ENABLE_TUNER_IIC, 0x01); /* open i2c bus switch */ nxt6000_write(i2c, demod_addr, ENABLE_TUNER_IIC, 0x01); /* open i2c bus switch */
ret = i2c->xfer(i2c, &msg, 1); ret = i2c_transfer(i2c, &msg, 1);
nxt6000_write(i2c, demod_addr, ENABLE_TUNER_IIC, 0x00); /* close i2c bus switch */ nxt6000_write(i2c, demod_addr, ENABLE_TUNER_IIC, 0x00); /* close i2c bus switch */
return (ret != 1) ? -EFAULT : 0; return (ret != 1) ? -EFAULT : 0;
} }
static int pll_write(struct dvb_i2c_bus *i2c, u8 demod_addr, u8 tuner_addr, u8 *buf, u8 len) static int pll_write(struct i2c_adapter *i2c, u8 demod_addr, u8 tuner_addr, u8 * buf, u8 len)
{ {
struct i2c_msg msg = {.addr = tuner_addr >> 1, .flags = 0, .buf = buf, .len = len}; struct i2c_msg msg = {.addr = tuner_addr >> 1, .flags = 0, .buf = buf, .len = len};
int ret; int ret;
nxt6000_write(i2c, demod_addr, ENABLE_TUNER_IIC, 0x01); /* open i2c bus switch */ nxt6000_write(i2c, demod_addr, ENABLE_TUNER_IIC, 0x01); /* open i2c bus switch */
ret = i2c->xfer(i2c, &msg, 1); ret = i2c_transfer(i2c, &msg, 1);
nxt6000_write(i2c, demod_addr, ENABLE_TUNER_IIC, 0x00); /* close i2c bus switch */ nxt6000_write(i2c, demod_addr, ENABLE_TUNER_IIC, 0x00); /* close i2c bus switch */
if (ret != 1) if (ret != 1)
dprintk("nxt6000: pll_write error %d\n", ret); dprintk("nxt6000: pll_write error %d\n", ret);
return (ret != 1) ? -EFAULT : 0; return (ret != 1) ? -EFAULT : 0;
} }
static int sp5659_set_tv_freq(struct dvb_frontend *fe, u32 freq) static int sp5659_set_tv_freq(struct nxt6000_config *nxt, u32 freq)
{ {
u8 buf[4]; u8 buf[4];
struct nxt6000_config *nxt = FE2NXT(fe);
buf[0] = (FREQ2DIV(freq) >> 8) & 0x7F; buf[0] = (FREQ2DIV(freq) >> 8) & 0x7F;
buf[1] = FREQ2DIV(freq) & 0xFF; buf[1] = FREQ2DIV(freq) & 0xFF;
...@@ -179,15 +166,12 @@ static int sp5659_set_tv_freq(struct dvb_frontend *fe, u32 freq) ...@@ -179,15 +166,12 @@ static int sp5659_set_tv_freq(struct dvb_frontend *fe, u32 freq)
else else
return -EINVAL; return -EINVAL;
return pll_write(fe->i2c, nxt->demod_addr, nxt->tuner_addr, buf, 4); return pll_write(nxt->i2c, nxt->demod_addr, nxt->tuner_addr, buf, 4);
} }
static int alp510_set_tv_freq(struct dvb_frontend *fe, u32 freq) static int alp510_set_tv_freq(struct nxt6000_config *nxt, u32 freq)
{ {
u8 buf[4]; u8 buf[4];
struct nxt6000_config *nxt = FE2NXT(fe);
buf[0] = (FREQ2DIV(freq) >> 8) & 0x7F; buf[0] = (FREQ2DIV(freq) >> 8) & 0x7F;
buf[1] = FREQ2DIV(freq) & 0xFF; buf[1] = FREQ2DIV(freq) & 0xFF;
...@@ -217,15 +201,12 @@ static int alp510_set_tv_freq(struct dvb_frontend *fe, u32 freq) ...@@ -217,15 +201,12 @@ static int alp510_set_tv_freq(struct dvb_frontend *fe, u32 freq)
return -EINVAL; return -EINVAL;
#endif #endif
return pll_write(fe->i2c, nxt->demod_addr, nxt->tuner_addr, buf, 4); return pll_write(nxt->i2c, nxt->demod_addr, nxt->tuner_addr, buf, 4);
} }
static int sp5730_set_tv_freq(struct dvb_frontend *fe, u32 freq) static int sp5730_set_tv_freq(struct nxt6000_config *nxt, u32 freq)
{ {
u8 buf[4]; u8 buf[4];
struct nxt6000_config *nxt = FE2NXT(fe);
buf[0] = (FREQ2DIV(freq) >> 8) & 0x7F; buf[0] = (FREQ2DIV(freq) >> 8) & 0x7F;
buf[1] = FREQ2DIV(freq) & 0xFF; buf[1] = FREQ2DIV(freq) & 0xFF;
...@@ -250,25 +231,21 @@ static int sp5730_set_tv_freq(struct dvb_frontend *fe, u32 freq) ...@@ -250,25 +231,21 @@ static int sp5730_set_tv_freq(struct dvb_frontend *fe, u32 freq)
else else
return -EINVAL; return -EINVAL;
return pll_write(fe->i2c, nxt->demod_addr, nxt->tuner_addr, buf, 4); return pll_write(nxt->i2c, nxt->demod_addr, nxt->tuner_addr, buf, 4);
} }
static void nxt6000_reset(struct dvb_frontend *fe) static void nxt6000_reset(struct nxt6000_config *fe)
{ {
u8 val; u8 val;
val = nxt6000_readreg(fe, OFDM_COR_CTL); val = nxt6000_readreg(fe, OFDM_COR_CTL);
nxt6000_writereg(fe, OFDM_COR_CTL, val & ~COREACT); nxt6000_writereg(fe, OFDM_COR_CTL, val & ~COREACT);
nxt6000_writereg(fe, OFDM_COR_CTL, val | COREACT); nxt6000_writereg(fe, OFDM_COR_CTL, val | COREACT);
} }
static int nxt6000_set_bandwidth(struct dvb_frontend *fe, fe_bandwidth_t bandwidth) static int nxt6000_set_bandwidth(struct nxt6000_config *fe, fe_bandwidth_t bandwidth)
{ {
u16 nominal_rate; u16 nominal_rate;
int result; int result;
...@@ -302,12 +279,10 @@ static int nxt6000_set_bandwidth(struct dvb_frontend *fe, fe_bandwidth_t bandwid ...@@ -302,12 +279,10 @@ static int nxt6000_set_bandwidth(struct dvb_frontend *fe, fe_bandwidth_t bandwid
return result; return result;
return nxt6000_writereg(fe, OFDM_TRL_NOMINALRATE_2, (nominal_rate >> 8) & 0xFF); return nxt6000_writereg(fe, OFDM_TRL_NOMINALRATE_2, (nominal_rate >> 8) & 0xFF);
} }
static int nxt6000_set_guard_interval(struct dvb_frontend *fe, fe_guard_interval_t guard_interval) static int nxt6000_set_guard_interval(struct nxt6000_config *fe, fe_guard_interval_t guard_interval)
{ {
switch(guard_interval) { switch(guard_interval) {
case GUARD_INTERVAL_1_32: case GUARD_INTERVAL_1_32:
...@@ -328,16 +303,12 @@ static int nxt6000_set_guard_interval(struct dvb_frontend *fe, fe_guard_interval ...@@ -328,16 +303,12 @@ static int nxt6000_set_guard_interval(struct dvb_frontend *fe, fe_guard_interval
return nxt6000_writereg(fe, OFDM_COR_MODEGUARD, 0x03 | (nxt6000_readreg(fe, OFDM_COR_MODEGUARD) & ~0x03)); return nxt6000_writereg(fe, OFDM_COR_MODEGUARD, 0x03 | (nxt6000_readreg(fe, OFDM_COR_MODEGUARD) & ~0x03));
default: default:
return -EINVAL; return -EINVAL;
} }
} }
static int nxt6000_set_inversion(struct dvb_frontend *fe, fe_spectral_inversion_t inversion) static int nxt6000_set_inversion(struct nxt6000_config *fe, fe_spectral_inversion_t inversion)
{ {
switch(inversion) { switch(inversion) {
case INVERSION_OFF: case INVERSION_OFF:
...@@ -353,12 +324,10 @@ static int nxt6000_set_inversion(struct dvb_frontend *fe, fe_spectral_inversion_ ...@@ -353,12 +324,10 @@ static int nxt6000_set_inversion(struct dvb_frontend *fe, fe_spectral_inversion_
return -EINVAL; return -EINVAL;
} }
} }
static int nxt6000_set_transmission_mode(struct dvb_frontend *fe, fe_transmit_mode_t transmission_mode) static int nxt6000_set_transmission_mode(struct nxt6000_config *fe, fe_transmit_mode_t transmission_mode)
{ {
int result; int result;
switch(transmission_mode) { switch(transmission_mode) {
...@@ -383,14 +352,10 @@ static int nxt6000_set_transmission_mode(struct dvb_frontend *fe, fe_transmit_mo ...@@ -383,14 +352,10 @@ static int nxt6000_set_transmission_mode(struct dvb_frontend *fe, fe_transmit_mo
return -EINVAL; return -EINVAL;
} }
} }
static void nxt6000_setup(struct dvb_frontend *fe) static void nxt6000_setup(struct nxt6000_config *fe)
{ {
struct nxt6000_config *nxt = FE2NXT(fe);
nxt6000_writereg(fe, RS_COR_SYNC_PARAM, SYNC_PARAM); nxt6000_writereg(fe, RS_COR_SYNC_PARAM, SYNC_PARAM);
nxt6000_writereg(fe, BER_CTRL, /*(1 << 2) |*/ (0x01 << 1) | 0x01); nxt6000_writereg(fe, BER_CTRL, /*(1 << 2) |*/ (0x01 << 1) | 0x01);
nxt6000_writereg(fe, VIT_COR_CTL, VIT_COR_RESYNC); nxt6000_writereg(fe, VIT_COR_CTL, VIT_COR_RESYNC);
...@@ -409,16 +374,15 @@ static void nxt6000_setup(struct dvb_frontend *fe) ...@@ -409,16 +374,15 @@ static void nxt6000_setup(struct dvb_frontend *fe)
nxt6000_writereg(fe, EN_DMD_RACQ, (1 << 7) | (3 << 4) | 2); nxt6000_writereg(fe, EN_DMD_RACQ, (1 << 7) | (3 << 4) | 2);
nxt6000_writereg(fe, DIAG_CONFIG, TB_SET); nxt6000_writereg(fe, DIAG_CONFIG, TB_SET);
if (nxt->clock_inversion) if (fe->clock_inversion)
nxt6000_writereg(fe, SUB_DIAG_MODE_SEL, CLKINVERSION); nxt6000_writereg(fe, SUB_DIAG_MODE_SEL, CLKINVERSION);
else else
nxt6000_writereg(fe, SUB_DIAG_MODE_SEL, 0); nxt6000_writereg(fe, SUB_DIAG_MODE_SEL, 0);
nxt6000_writereg(fe, TS_FORMAT, 0); nxt6000_writereg(fe, TS_FORMAT, 0);
} }
static void nxt6000_dump_status(struct dvb_frontend *fe) static void nxt6000_dump_status(struct nxt6000_config *fe)
{ {
u8 val; u8 val;
...@@ -673,13 +637,12 @@ static void nxt6000_dump_status(struct dvb_frontend *fe) ...@@ -673,13 +637,12 @@ static void nxt6000_dump_status(struct dvb_frontend *fe)
val = nxt6000_readreg(fe, RF_AGC_STATUS); val = nxt6000_readreg(fe, RF_AGC_STATUS);
printk(" RF AGC LOCK: %d,", (val >> 4) & 0x01); printk(" RF AGC LOCK: %d,", (val >> 4) & 0x01);
printk("\n"); printk("\n");
} }
static int nxt6000_ioctl(struct dvb_frontend *fe, unsigned int cmd, void *arg) static int nxt6000_ioctl(struct dvb_frontend *f, unsigned int cmd, void *arg)
{ {
struct nxt6000_config *fe = (struct nxt6000_config *) f->data;
switch (cmd) { switch (cmd) {
...@@ -769,17 +732,14 @@ static int nxt6000_ioctl(struct dvb_frontend *fe, unsigned int cmd, void *arg) ...@@ -769,17 +732,14 @@ static int nxt6000_ioctl(struct dvb_frontend *fe, unsigned int cmd, void *arg)
case FE_SET_FRONTEND: case FE_SET_FRONTEND:
{ {
struct nxt6000_config *nxt = FE2NXT(fe);
struct dvb_frontend_parameters *param = (struct dvb_frontend_parameters *)arg; struct dvb_frontend_parameters *param = (struct dvb_frontend_parameters *)arg;
int result; int result;
switch(nxt->tuner_type) { switch (fe->tuner_type) {
case TUNER_TYPE_ALP510: case TUNER_TYPE_ALP510:
if ((result = alp510_set_tv_freq(fe, param->frequency)) < 0) if ((result = alp510_set_tv_freq(fe, param->frequency)) < 0)
return result; return result;
break; break;
case TUNER_TYPE_SP5659: case TUNER_TYPE_SP5659:
...@@ -826,42 +786,41 @@ static int nxt6000_ioctl(struct dvb_frontend *fe, unsigned int cmd, void *arg) ...@@ -826,42 +786,41 @@ static int nxt6000_ioctl(struct dvb_frontend *fe, unsigned int cmd, void *arg)
static u8 demod_addr_tbl[] = {0x14, 0x18, 0x24, 0x28}; static u8 demod_addr_tbl[] = {0x14, 0x18, 0x24, 0x28};
static int nxt6000_attach(struct dvb_i2c_bus *i2c, void **data) static struct i2c_client client_template;
static int attach_adapter(struct i2c_adapter *adapter)
{ {
struct i2c_client *client;
struct nxt6000_config *nxt;
u8 addr_nr; u8 addr_nr;
u8 fe_count = 0; int ret;
struct nxt6000_config *pnxt;
dprintk("nxt6000: attach\n");
pnxt = kmalloc(sizeof(demod_addr_tbl)*sizeof(struct nxt6000_config), GFP_KERNEL); if ((nxt = kmalloc(sizeof(struct nxt6000_config), GFP_KERNEL)) == NULL)
if (NULL == pnxt) {
dprintk("nxt6000: no memory for private data.\n");
return -ENOMEM; return -ENOMEM;
}
*data = pnxt; memset(nxt, 0, sizeof(*nxt));
nxt->i2c = adapter;
for (addr_nr = 0; addr_nr < sizeof(demod_addr_tbl); addr_nr++) { for (addr_nr = 0; addr_nr < sizeof(demod_addr_tbl); addr_nr++) {
struct nxt6000_config *nxt = &pnxt[addr_nr];
if (nxt6000_read(i2c, demod_addr_tbl[addr_nr], OFDM_MSC_REV) != NXT6000ASICDEVICE) if (nxt6000_read(adapter, demod_addr_tbl[addr_nr], OFDM_MSC_REV) != NXT6000ASICDEVICE)
continue; continue;
if (pll_test(i2c, demod_addr_tbl[addr_nr], 0xC0) == 0) { if (pll_test(adapter, demod_addr_tbl[addr_nr], 0xC0) == 0) {
nxt->tuner_addr = 0xC0; nxt->tuner_addr = 0xC0;
nxt->tuner_type = TUNER_TYPE_ALP510; nxt->tuner_type = TUNER_TYPE_ALP510;
nxt->clock_inversion = 1; nxt->clock_inversion = 1;
dprintk("nxt6000: detected TI ALP510 tuner at 0x%02X\n", nxt->tuner_addr); dprintk("nxt6000: detected TI ALP510 tuner at 0x%02X\n", nxt->tuner_addr);
} else if (pll_test(i2c, demod_addr_tbl[addr_nr], 0xC2) == 0) { } else if (pll_test(adapter, demod_addr_tbl[addr_nr], 0xC2) == 0) {
nxt->tuner_addr = 0xC2; nxt->tuner_addr = 0xC2;
nxt->tuner_type = TUNER_TYPE_SP5659; nxt->tuner_type = TUNER_TYPE_SP5659;
nxt->clock_inversion = 0; nxt->clock_inversion = 0;
dprintk("nxt6000: detected MITEL SP5659 tuner at 0x%02X\n", nxt->tuner_addr); dprintk("nxt6000: detected MITEL SP5659 tuner at 0x%02X\n", nxt->tuner_addr);
} else if (pll_test(i2c, demod_addr_tbl[addr_nr], 0xC0) == 0) { } else if (pll_test(adapter, demod_addr_tbl[addr_nr], 0xC0) == 0) {
nxt->tuner_addr = 0xC0; nxt->tuner_addr = 0xC0;
nxt->tuner_type = TUNER_TYPE_SP5730; nxt->tuner_type = TUNER_TYPE_SP5730;
nxt->clock_inversion = 0; nxt->clock_inversion = 0;
...@@ -872,48 +831,99 @@ static int nxt6000_attach(struct dvb_i2c_bus *i2c, void **data) ...@@ -872,48 +831,99 @@ static int nxt6000_attach(struct dvb_i2c_bus *i2c, void **data)
printk("nxt6000: unable to detect tuner\n"); printk("nxt6000: unable to detect tuner\n");
continue; continue;
} }
nxt->demod_addr = demod_addr_tbl[addr_nr];
dprintk("nxt6000: attached at %d:%d\n", i2c->adapter->num, i2c->id);
dvb_register_frontend(nxt6000_ioctl, i2c, (void *)nxt, &nxt6000_info);
fe_count++;
} }
if (fe_count == 0) { if (addr_nr == sizeof(demod_addr_tbl)) {
kfree(pnxt); kfree(nxt);
return -ENODEV; return -ENODEV;
} }
return 0; nxt->demod_addr = demod_addr_tbl[addr_nr];
if (NULL == (client = kmalloc(sizeof(struct i2c_client), GFP_KERNEL))) {
kfree(nxt);
return -ENOMEM;
}
memcpy(client, &client_template, sizeof(struct i2c_client));
client->adapter = adapter;
client->addr = demod_addr_tbl[addr_nr];
i2c_set_clientdata(client, (void *) nxt);
ret = i2c_attach_client(client);
if (ret)
goto out;
BUG_ON(!nxt->dvb);
ret = dvb_register_frontend(nxt6000_ioctl, nxt->dvb, nxt, &nxt6000_info, THIS_MODULE);
if (ret) {
i2c_detach_client(client);
goto out;
}
ret = 0;
out:
kfree(client);
kfree(nxt);
return ret;
} }
static void nxt6000_detach(struct dvb_i2c_bus *i2c, void *data) static int detach_client(struct i2c_client *client)
{ {
struct nxt6000_config *pnxt = (struct nxt6000_config *)data; struct nxt6000_config *state = (struct nxt6000_config *) i2c_get_clientdata(client);
dprintk("nxt6000: detach\n"); dvb_unregister_frontend_new(nxt6000_ioctl, state->dvb);
dvb_unregister_frontend(nxt6000_ioctl, i2c); i2c_detach_client(client);
kfree(pnxt); BUG_ON(state->dvb);
kfree(client);
kfree(state);
return 0;
} }
static __init int nxt6000_init(void) static int command(struct i2c_client *client, unsigned int cmd, void *arg)
{ {
struct nxt6000_config *state = (struct nxt6000_config *) i2c_get_clientdata(client);
dprintk("nxt6000: init\n"); switch (cmd) {
case FE_REGISTER:{
state->dvb = (struct dvb_adapter *) arg;
break;
}
case FE_UNREGISTER:{
state->dvb = NULL;
break;
}
default:
return -EOPNOTSUPP;
}
return 0;
}
return dvb_register_i2c_device(THIS_MODULE, nxt6000_attach, nxt6000_detach); static struct i2c_driver driver = {
.owner = THIS_MODULE,
.name = "nxt6000",
.id = I2C_DRIVERID_DVBFE_NXT6000,
.flags = I2C_DF_NOTIFY,
.attach_adapter = attach_adapter,
.detach_client = detach_client,
.command = command,
};
static struct i2c_client client_template = {
I2C_DEVNAME("nxt6000"),
.flags = I2C_CLIENT_ALLOW_USE,
.driver = &driver,
};
static __init int nxt6000_init(void)
{
return i2c_add_driver(&driver);
} }
static __exit void nxt6000_exit(void) static __exit void nxt6000_exit(void)
{ {
if (i2c_del_driver(&driver))
dprintk("nxt6000: cleanup\n"); printk("nxt6000: driver deregistration failed\n");
dvb_unregister_i2c_device(nxt6000_attach);
} }
module_init(nxt6000_init); module_init(nxt6000_init);
......
...@@ -3,42 +3,31 @@ ...@@ -3,42 +3,31 @@
*/ */
/* /*
This driver needs a copy of the Avermedia firmware. The version tested * This driver needs external firmware. Please use the command
is part of the Avermedia DVB-T 1.3.26.3 Application. If the software is * "<kerneldir>/Documentation/dvb/get_dvb_firmware sp887x" to
installed in Windows the file will be in the /Program Files/AVerTV DVB-T/ * download/extract it, and then copy it to /usr/lib/hotplug/firmware.
directory and is called sc_main.mc. Alternatively it can "extracted" from
the install cab files. Copy this file to '/usr/lib/hotplug/firmware/sc_main.mc'.
With this version of the file the first 10 bytes are discarded and the
next 0x4000 loaded. This may change in future versions.
*/ */
#define SP887X_DEFAULT_FIRMWARE "dvb-fe-sp887x.fw"
#include <linux/kernel.h>
#include <linux/vmalloc.h>
#include <linux/module.h>
#include <linux/init.h> #include <linux/init.h>
#include <linux/string.h> #include <linux/module.h>
#include <linux/slab.h> #include <linux/moduleparam.h>
#include <linux/fs.h> #include <linux/device.h>
#include <linux/unistd.h> #include <linux/firmware.h>
#include <linux/fcntl.h>
#include <linux/errno.h>
#include <linux/i2c.h>
#include <linux/syscalls.h>
#include "dvb_frontend.h" #include "dvb_frontend.h"
#include "dvb_functions.h"
#ifndef DVB_SP887X_FIRMWARE_FILE #define FRONTEND_NAME "dvbfe_sp887x"
#define DVB_SP887X_FIRMWARE_FILE "/usr/lib/hotplug/firmware/sc_main.mc"
#endif
static char *sp887x_firmware = DVB_SP887X_FIRMWARE_FILE; #define dprintk(args...) \
do { \
if (debug) printk(KERN_DEBUG FRONTEND_NAME ": " args); \
} while (0)
#if 0 static int debug;
#define dprintk(x...) printk(x)
#else module_param(debug, int, 0644);
#define dprintk(x...) MODULE_PARM_DESC(debug, "Turn on/off frontend debugging (default:off).");
#endif
#if 0 #if 0
#define LOG(dir,addr,buf,len) \ #define LOG(dir,addr,buf,len) \
...@@ -53,9 +42,7 @@ static char *sp887x_firmware = DVB_SP887X_FIRMWARE_FILE; ...@@ -53,9 +42,7 @@ static char *sp887x_firmware = DVB_SP887X_FIRMWARE_FILE;
#define LOG(dir,addr,buf,len) #define LOG(dir,addr,buf,len)
#endif #endif
static struct dvb_frontend_info sp887x_info = {
static
struct dvb_frontend_info sp887x_info = {
.name = "Microtune MT7202DTF", .name = "Microtune MT7202DTF",
.type = FE_OFDM, .type = FE_OFDM,
.frequency_min = 50500000, .frequency_min = 50500000,
...@@ -67,16 +54,19 @@ struct dvb_frontend_info sp887x_info = { ...@@ -67,16 +54,19 @@ struct dvb_frontend_info sp887x_info = {
FE_CAN_RECOVER FE_CAN_RECOVER
}; };
static struct sp887x_state {
int i2c_writebytes (struct dvb_frontend *fe, u8 addr, u8 *buf, u8 len) struct i2c_adapter *i2c;
struct dvb_adapter *dvb;
};
static int i2c_writebytes (struct i2c_adapter *i2c, u8 addr, u8 *buf, u8 len)
{ {
struct dvb_i2c_bus *i2c = fe->i2c;
struct i2c_msg msg = { .addr = addr, .flags = 0, .buf = buf, .len = len }; struct i2c_msg msg = { .addr = addr, .flags = 0, .buf = buf, .len = len };
int err; int err;
LOG("i2c_writebytes", msg.addr, msg.buf, msg.len); LOG("i2c_writebytes", msg.addr, msg.buf, msg.len);
if ((err = i2c->xfer (i2c, &msg, 1)) != 1) { if ((err = i2c_transfer (i2c, &msg, 1)) != 1) {
printk ("%s: i2c write error (addr %02x, err == %i)\n", printk ("%s: i2c write error (addr %02x, err == %i)\n",
__FUNCTION__, addr, err); __FUNCTION__, addr, err);
return -EREMOTEIO; return -EREMOTEIO;
...@@ -85,23 +75,19 @@ int i2c_writebytes (struct dvb_frontend *fe, u8 addr, u8 *buf, u8 len) ...@@ -85,23 +75,19 @@ int i2c_writebytes (struct dvb_frontend *fe, u8 addr, u8 *buf, u8 len)
return 0; return 0;
} }
static int sp887x_writereg (struct i2c_adapter *i2c, u16 reg, u16 data)
static
int sp887x_writereg (struct dvb_frontend *fe, u16 reg, u16 data)
{ {
struct dvb_i2c_bus *i2c = fe->i2c;
u8 b0 [] = { reg >> 8 , reg & 0xff, data >> 8, data & 0xff }; u8 b0 [] = { reg >> 8 , reg & 0xff, data >> 8, data & 0xff };
struct i2c_msg msg = { .addr = 0x70, .flags = 0, .buf = b0, .len = 4 }; struct i2c_msg msg = { .addr = 0x70, .flags = 0, .buf = b0, .len = 4 };
int ret; int ret;
LOG("sp887x_writereg", msg.addr, msg.buf, msg.len); LOG("sp887x_writereg", msg.addr, msg.buf, msg.len);
if ((ret = i2c->xfer(i2c, &msg, 1)) != 1) { if ((ret = i2c_transfer(i2c, &msg, 1)) != 1) {
/** /**
* in case of soft reset we ignore ACK errors... * in case of soft reset we ignore ACK errors...
*/ */
if (!(reg == 0xf1a && data == 0x000 && if (!(reg == 0xf1a && data == 0x000 &&
(ret == -EREMOTEIO || ret == -EFAULT))) (ret == -EREMOTEIO || ret == -EFAULT)))
{ {
printk("%s: writereg error " printk("%s: writereg error "
...@@ -114,11 +100,8 @@ int sp887x_writereg (struct dvb_frontend *fe, u16 reg, u16 data) ...@@ -114,11 +100,8 @@ int sp887x_writereg (struct dvb_frontend *fe, u16 reg, u16 data)
return 0; return 0;
} }
static u16 sp887x_readreg (struct i2c_adapter *i2c, u16 reg)
static
u16 sp887x_readreg (struct dvb_frontend *fe, u16 reg)
{ {
struct dvb_i2c_bus *i2c = fe->i2c;
u8 b0 [] = { reg >> 8 , reg & 0xff }; u8 b0 [] = { reg >> 8 , reg & 0xff };
u8 b1 [2]; u8 b1 [2];
int ret; int ret;
...@@ -128,39 +111,33 @@ u16 sp887x_readreg (struct dvb_frontend *fe, u16 reg) ...@@ -128,39 +111,33 @@ u16 sp887x_readreg (struct dvb_frontend *fe, u16 reg)
LOG("sp887x_readreg (w)", msg[0].addr, msg[0].buf, msg[0].len); LOG("sp887x_readreg (w)", msg[0].addr, msg[0].buf, msg[0].len);
LOG("sp887x_readreg (r)", msg[1].addr, msg[1].buf, msg[1].len); LOG("sp887x_readreg (r)", msg[1].addr, msg[1].buf, msg[1].len);
if ((ret = i2c->xfer(i2c, msg, 2)) != 2) if ((ret = i2c_transfer(i2c, msg, 2)) != 2)
printk("%s: readreg error (ret == %i)\n", __FUNCTION__, ret); printk("%s: readreg error (ret == %i)\n", __FUNCTION__, ret);
return (((b1[0] << 8) | b1[1]) & 0xfff); return (((b1[0] << 8) | b1[1]) & 0xfff);
} }
static void sp887x_microcontroller_stop (struct i2c_adapter *fe)
static
void sp887x_microcontroller_stop (struct dvb_frontend *fe)
{ {
dprintk("%s\n", __FUNCTION__); dprintk("%s\n", __FUNCTION__);
sp887x_writereg(fe, 0xf08, 0x000); sp887x_writereg(fe, 0xf08, 0x000);
sp887x_writereg(fe, 0xf09, 0x000); sp887x_writereg(fe, 0xf09, 0x000);
/* microcontroller STOP */ /* microcontroller STOP */
sp887x_writereg(fe, 0xf00, 0x000); sp887x_writereg(fe, 0xf00, 0x000);
} }
static void sp887x_microcontroller_start (struct i2c_adapter *fe)
static
void sp887x_microcontroller_start (struct dvb_frontend *fe)
{ {
dprintk("%s\n", __FUNCTION__); dprintk("%s\n", __FUNCTION__);
sp887x_writereg(fe, 0xf08, 0x000); sp887x_writereg(fe, 0xf08, 0x000);
sp887x_writereg(fe, 0xf09, 0x000); sp887x_writereg(fe, 0xf09, 0x000);
/* microcontroller START */ /* microcontroller START */
sp887x_writereg(fe, 0xf00, 0x001); sp887x_writereg(fe, 0xf00, 0x001);
} }
static void sp887x_setup_agc (struct i2c_adapter *fe)
static
void sp887x_setup_agc (struct dvb_frontend *fe)
{ {
/* setup AGC parameters */ /* setup AGC parameters */
dprintk("%s\n", __FUNCTION__); dprintk("%s\n", __FUNCTION__);
...@@ -180,72 +157,31 @@ void sp887x_setup_agc (struct dvb_frontend *fe) ...@@ -180,72 +157,31 @@ void sp887x_setup_agc (struct dvb_frontend *fe)
sp887x_writereg(fe, 0x303, 0x000); sp887x_writereg(fe, 0x303, 0x000);
} }
#define BLOCKSIZE 30 #define BLOCKSIZE 30
#define FW_SIZE 0x4000
/** /**
* load firmware and setup MPEG interface... * load firmware and setup MPEG interface...
*/ */
static static int sp887x_initial_setup (struct i2c_adapter *fe, const struct firmware *fw)
int sp887x_initial_setup (struct dvb_frontend *fe)
{ {
u8 buf [BLOCKSIZE+2]; u8 buf [BLOCKSIZE+2];
unsigned char *firmware = NULL;
int i; int i;
int fd; int fw_size = fw->size;
int filesize; unsigned char *mem = fw->data;
int fw_size;
mm_segment_t fs;
dprintk("%s\n", __FUNCTION__); dprintk("%s\n", __FUNCTION__);
/* ignore the first 10 bytes, then we expect 0x4000 bytes of firmware */
if (fw_size < FW_SIZE+10)
return -ENODEV;
mem = fw->data + 10;
/* soft reset */ /* soft reset */
sp887x_writereg(fe, 0xf1a, 0x000); sp887x_writereg(fe, 0xf1a, 0x000);
sp887x_microcontroller_stop (fe); sp887x_microcontroller_stop (fe);
fs = get_fs();
// Load the firmware
set_fs(get_ds());
fd = sys_open(sp887x_firmware, 0, 0);
if (fd < 0) {
printk(KERN_WARNING "%s: Unable to open firmware %s\n", __FUNCTION__,
sp887x_firmware);
return -EIO;
}
filesize = sys_lseek(fd, 0L, 2);
if (filesize <= 0) {
printk(KERN_WARNING "%s: Firmware %s is empty\n", __FUNCTION__,
sp887x_firmware);
sys_close(fd);
return -EIO;
}
fw_size = 0x4000;
// allocate buffer for it
firmware = vmalloc(fw_size);
if (firmware == NULL) {
printk(KERN_WARNING "%s: Out of memory loading firmware\n",
__FUNCTION__);
sys_close(fd);
return -EIO;
}
// read it!
// read the first 16384 bytes from the file
// ignore the first 10 bytes
sys_lseek(fd, 10, 0);
if (sys_read(fd, firmware, fw_size) != fw_size) {
printk(KERN_WARNING "%s: Failed to read firmware\n", __FUNCTION__);
vfree(firmware);
sys_close(fd);
return -EIO;
}
sys_close(fd);
set_fs(fs);
printk ("%s: firmware upload... ", __FUNCTION__); printk ("%s: firmware upload... ", __FUNCTION__);
/* setup write pointer to -1 (end of memory) */ /* setup write pointer to -1 (end of memory) */
...@@ -255,12 +191,12 @@ int sp887x_initial_setup (struct dvb_frontend *fe) ...@@ -255,12 +191,12 @@ int sp887x_initial_setup (struct dvb_frontend *fe)
/* dummy write (wrap around to start of memory) */ /* dummy write (wrap around to start of memory) */
sp887x_writereg(fe, 0x8f0a, 0x0000); sp887x_writereg(fe, 0x8f0a, 0x0000);
for (i=0; i<fw_size; i+=BLOCKSIZE) { for (i = 0; i < FW_SIZE; i += BLOCKSIZE) {
int c = BLOCKSIZE; int c = BLOCKSIZE;
int err; int err;
if (i+c > fw_size) if (i+c > FW_SIZE)
c = fw_size - i; c = FW_SIZE - i;
/* bit 0x8000 in address is set to enable 13bit mode */ /* bit 0x8000 in address is set to enable 13bit mode */
/* bit 0x4000 enables multibyte read/write transfers */ /* bit 0x4000 enables multibyte read/write transfers */
...@@ -268,18 +204,15 @@ int sp887x_initial_setup (struct dvb_frontend *fe) ...@@ -268,18 +204,15 @@ int sp887x_initial_setup (struct dvb_frontend *fe)
buf[0] = 0xcf; buf[0] = 0xcf;
buf[1] = 0x0a; buf[1] = 0x0a;
memcpy(&buf[2], firmware + i, c); memcpy(&buf[2], mem + i, c);
if ((err = i2c_writebytes (fe, 0x70, buf, c+2)) < 0) { if ((err = i2c_writebytes (fe, 0x70, buf, c+2)) < 0) {
printk ("failed.\n"); printk ("failed.\n");
printk ("%s: i2c error (err == %i)\n", __FUNCTION__, err); printk ("%s: i2c error (err == %i)\n", __FUNCTION__, err);
vfree(firmware);
return err; return err;
} }
} }
vfree(firmware);
/* don't write RS bytes between packets */ /* don't write RS bytes between packets */
sp887x_writereg(fe, 0xc13, 0x001); sp887x_writereg(fe, 0xc13, 0x001);
...@@ -305,13 +238,11 @@ int sp887x_initial_setup (struct dvb_frontend *fe) ...@@ -305,13 +238,11 @@ int sp887x_initial_setup (struct dvb_frontend *fe)
return 0; return 0;
}; };
/** /**
* returns the actual tuned center frequency which can be used * returns the actual tuned center frequency which can be used
* to initialise the AFC registers * to initialise the AFC registers
*/ */
static static int tsa5060_setup_pll (struct i2c_adapter *fe, int freq)
int tsa5060_setup_pll (struct dvb_frontend *fe, int freq)
{ {
u8 cfg, cpump, band_select; u8 cfg, cpump, band_select;
u8 buf [4]; u8 buf [4];
...@@ -338,13 +269,10 @@ int tsa5060_setup_pll (struct dvb_frontend *fe, int freq) ...@@ -338,13 +269,10 @@ int tsa5060_setup_pll (struct dvb_frontend *fe, int freq)
return (div * 166666 - 36000000); return (div * 166666 - 36000000);
} }
static int configure_reg0xc05 (struct dvb_frontend_parameters *p, u16 *reg0xc05)
static
int configure_reg0xc05 (struct dvb_frontend_parameters *p, u16 *reg0xc05)
{ {
int known_parameters = 1; int known_parameters = 1;
*reg0xc05 = 0x000; *reg0xc05 = 0x000;
switch (p->u.ofdm.constellation) { switch (p->u.ofdm.constellation) {
...@@ -412,13 +340,11 @@ int configure_reg0xc05 (struct dvb_frontend_parameters *p, u16 *reg0xc05) ...@@ -412,13 +340,11 @@ int configure_reg0xc05 (struct dvb_frontend_parameters *p, u16 *reg0xc05)
return 0; return 0;
} }
/** /**
* estimates division of two 24bit numbers, * estimates division of two 24bit numbers,
* derived from the ves1820/stv0299 driver code * derived from the ves1820/stv0299 driver code
*/ */
static static void divide (int n, int d, int *quotient_i, int *quotient_f)
void divide (int n, int d, int *quotient_i, int *quotient_f)
{ {
unsigned int q, r; unsigned int q, r;
...@@ -436,9 +362,7 @@ void divide (int n, int d, int *quotient_i, int *quotient_f) ...@@ -436,9 +362,7 @@ void divide (int n, int d, int *quotient_i, int *quotient_f)
} }
} }
static void sp887x_correct_offsets (struct i2c_adapter *fe,
static
void sp887x_correct_offsets (struct dvb_frontend *fe,
struct dvb_frontend_parameters *p, struct dvb_frontend_parameters *p,
int actual_freq) int actual_freq)
{ {
...@@ -469,9 +393,7 @@ void sp887x_correct_offsets (struct dvb_frontend *fe, ...@@ -469,9 +393,7 @@ void sp887x_correct_offsets (struct dvb_frontend *fe,
sp887x_writereg(fe, 0x30a, frequency_shift & 0xfff); sp887x_writereg(fe, 0x30a, frequency_shift & 0xfff);
} }
static int sp887x_setup_frontend_parameters (struct i2c_adapter *fe,
static
int sp887x_setup_frontend_parameters (struct dvb_frontend *fe,
struct dvb_frontend_parameters *p) struct dvb_frontend_parameters *p)
{ {
int actual_freq, err; int actual_freq, err;
...@@ -481,7 +403,7 @@ int sp887x_setup_frontend_parameters (struct dvb_frontend *fe, ...@@ -481,7 +403,7 @@ int sp887x_setup_frontend_parameters (struct dvb_frontend *fe,
p->u.ofdm.bandwidth != BANDWIDTH_7_MHZ && p->u.ofdm.bandwidth != BANDWIDTH_7_MHZ &&
p->u.ofdm.bandwidth != BANDWIDTH_6_MHZ) p->u.ofdm.bandwidth != BANDWIDTH_6_MHZ)
return -EINVAL; return -EINVAL;
if ((err = configure_reg0xc05(p, &reg0xc05))) if ((err = configure_reg0xc05(p, &reg0xc05)))
return err; return err;
...@@ -529,10 +451,11 @@ int sp887x_setup_frontend_parameters (struct dvb_frontend *fe, ...@@ -529,10 +451,11 @@ int sp887x_setup_frontend_parameters (struct dvb_frontend *fe,
return 0; return 0;
} }
static int sp887x_ioctl(struct dvb_frontend *f, unsigned int cmd, void *arg)
static
int sp887x_ioctl (struct dvb_frontend *fe, unsigned int cmd, void *arg)
{ {
struct sp887x_state *state = (struct sp887x_state *) f->data;
struct i2c_adapter *fe = state->i2c;
switch (cmd) { switch (cmd) {
case FE_GET_INFO: case FE_GET_INFO:
memcpy (arg, &sp887x_info, sizeof(struct dvb_frontend_info)); memcpy (arg, &sp887x_info, sizeof(struct dvb_frontend_info));
...@@ -622,10 +545,6 @@ int sp887x_ioctl (struct dvb_frontend *fe, unsigned int cmd, void *arg) ...@@ -622,10 +545,6 @@ int sp887x_ioctl (struct dvb_frontend *fe, unsigned int cmd, void *arg)
break; break;
case FE_INIT: case FE_INIT:
if (fe->data == NULL) { /* first time initialisation... */
fe->data = (void*) ~0;
sp887x_initial_setup (fe);
}
/* enable TS output and interface pins */ /* enable TS output and interface pins */
sp887x_writereg(fe, 0xc18, 0x00d); sp887x_writereg(fe, 0xc18, 0x00d);
break; break;
...@@ -633,11 +552,11 @@ int sp887x_ioctl (struct dvb_frontend *fe, unsigned int cmd, void *arg) ...@@ -633,11 +552,11 @@ int sp887x_ioctl (struct dvb_frontend *fe, unsigned int cmd, void *arg)
case FE_GET_TUNE_SETTINGS: case FE_GET_TUNE_SETTINGS:
{ {
struct dvb_frontend_tune_settings* fesettings = (struct dvb_frontend_tune_settings*) arg; struct dvb_frontend_tune_settings* fesettings = (struct dvb_frontend_tune_settings*) arg;
fesettings->min_delay_ms = 50; fesettings->min_delay_ms = 350;
fesettings->step_size = 0; fesettings->step_size = 166666*2;
fesettings->max_drift = 0; fesettings->max_drift = (166666*2)+1;
return 0; return 0;
} }
default: default:
return -EOPNOTSUPP; return -EOPNOTSUPP;
...@@ -646,51 +565,139 @@ int sp887x_ioctl (struct dvb_frontend *fe, unsigned int cmd, void *arg) ...@@ -646,51 +565,139 @@ int sp887x_ioctl (struct dvb_frontend *fe, unsigned int cmd, void *arg)
return 0; return 0;
} }
static struct i2c_client client_template;
static int attach_adapter(struct i2c_adapter *adapter)
static
int sp887x_attach (struct dvb_i2c_bus *i2c, void **data)
{ {
struct i2c_client *client;
struct sp887x_state *state;
const struct firmware *fw;
int ret;
struct i2c_msg msg = {.addr = 0x70, .flags = 0, .buf = NULL, .len = 0 }; struct i2c_msg msg = {.addr = 0x70, .flags = 0, .buf = NULL, .len = 0 };
dprintk ("%s\n", __FUNCTION__); dprintk ("%s\n", __FUNCTION__);
if (i2c->xfer (i2c, &msg, 1) != 1) if (NULL == (client = kmalloc(sizeof(struct i2c_client), GFP_KERNEL))) {
return -ENOMEM;
}
if (NULL == (state = kmalloc(sizeof(struct sp887x_state), GFP_KERNEL))) {
kfree(client);
return -ENOMEM;
}
state->i2c = adapter;
if (i2c_transfer (adapter, &msg, 1) != 1) {
kfree(state);
kfree(client);
return -ENODEV; return -ENODEV;
}
return dvb_register_frontend (sp887x_ioctl, i2c, NULL, &sp887x_info); memcpy(client, &client_template, sizeof(struct i2c_client));
} client->adapter = adapter;
i2c_set_clientdata(client, (void*)state);
ret = i2c_attach_client(client);
if (ret) {
kfree(client);
kfree(state);
return ret;
}
/* request the firmware, this will block until someone uploads it */
printk("sp887x: waiting for firmware upload...\n");
ret = request_firmware(&fw, SP887X_DEFAULT_FIRMWARE, &client->dev);
if (ret) {
printk("sp887x: no firmware upload (timeout or file not found?)\n");
goto out;
}
ret = sp887x_initial_setup(adapter, fw);
if (ret) {
printk("sp887x: writing firmware to device failed\n");
goto out;
}
ret = dvb_register_frontend(sp887x_ioctl, state->dvb, state,
&sp887x_info, THIS_MODULE);
if (ret) {
printk("sp887x: registering frontend to dvb-core failed.\n");
goto out;
}
return 0;
out:
release_firmware(fw);
i2c_detach_client(client);
kfree(client);
kfree(state);
return ret;
}
static static int detach_client(struct i2c_client *client)
void sp887x_detach (struct dvb_i2c_bus *i2c, void *data)
{ {
struct sp887x_state *state = (struct sp887x_state*)i2c_get_clientdata(client);
dprintk ("%s\n", __FUNCTION__); dprintk ("%s\n", __FUNCTION__);
dvb_unregister_frontend (sp887x_ioctl, i2c);
}
dvb_unregister_frontend (sp887x_ioctl, state->dvb);
i2c_detach_client(client);
BUG_ON(state->dvb);
kfree(client);
kfree(state);
return 0;
}
static static int command (struct i2c_client *client, unsigned int cmd, void *arg)
int __init init_sp887x (void)
{ {
struct sp887x_state *state = (struct sp887x_state*)i2c_get_clientdata(client);
dprintk ("%s\n", __FUNCTION__); dprintk ("%s\n", __FUNCTION__);
return dvb_register_i2c_device (NULL, sp887x_attach, sp887x_detach);
switch (cmd) {
case FE_REGISTER:
state->dvb = (struct dvb_adapter*)arg;
break;
case FE_UNREGISTER:
state->dvb = NULL;
break;
default:
return -EOPNOTSUPP;
}
return 0;
} }
static struct i2c_driver driver = {
.owner = THIS_MODULE,
.name = FRONTEND_NAME,
.id = I2C_DRIVERID_DVBFE_SP887X,
.flags = I2C_DF_NOTIFY,
.attach_adapter = attach_adapter,
.detach_client = detach_client,
.command = command,
};
static struct i2c_client client_template = {
.name = FRONTEND_NAME,
.flags = I2C_CLIENT_ALLOW_USE,
.driver = &driver,
};
static static int __init init_sp887x(void)
void __exit exit_sp887x (void)
{ {
dprintk ("%s\n", __FUNCTION__); return i2c_add_driver(&driver);
dvb_unregister_i2c_device (sp887x_attach);
} }
static void __exit exit_sp887x(void)
{
if (i2c_del_driver(&driver))
printk("sp887x: driver deregistration failed\n");
}
module_init(init_sp887x); module_init(init_sp887x);
module_exit(exit_sp887x); module_exit(exit_sp887x);
MODULE_DESCRIPTION("sp887x DVB-T demodulator driver"); MODULE_DESCRIPTION("sp887x DVB-T demodulator driver");
MODULE_LICENSE("GPL"); MODULE_LICENSE("GPL");
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment