#include "rtl28xxu.h"
 
-#include "rtl2830.h"
-#include "rtl2832.h"
-#include "rtl2832_sdr.h"
-#include "mn88472.h"
-#include "mn88473.h"
-
-#include "qt1010.h"
-#include "mt2060.h"
-#include "mxl5005s.h"
-#include "fc0012.h"
-#include "fc0013.h"
-#include "e4000.h"
-#include "fc2580.h"
-#include "tua9001.h"
-#include "r820t.h"
-
-
 #ifdef CONFIG_MEDIA_ATTACH
 #define dvb_attach_sdr(FUNCTION, ARGS...) ({ \
        void *__r = NULL; \
        return ret;
 }
 
-static const struct rtl2830_config rtl28xxu_rtl2830_mt2060_config = {
-       .i2c_addr = 0x10, /* 0x20 */
-       .xtal = 28800000,
-       .ts_mode = 0,
+static const struct rtl2830_platform_data rtl2830_mt2060_platform_data = {
+       .clk = 28800000,
        .spec_inv = 1,
        .vtop = 0x20,
        .krf = 0x04,
 
 };
 
-static const struct rtl2830_config rtl28xxu_rtl2830_qt1010_config = {
-       .i2c_addr = 0x10, /* 0x20 */
-       .xtal = 28800000,
-       .ts_mode = 0,
+static const struct rtl2830_platform_data rtl2830_qt1010_platform_data = {
+       .clk = 28800000,
        .spec_inv = 1,
        .vtop = 0x20,
        .krf = 0x04,
        .agc_targ_val = 0x2d,
 };
 
-static const struct rtl2830_config rtl28xxu_rtl2830_mxl5005s_config = {
-       .i2c_addr = 0x10, /* 0x20 */
-       .xtal = 28800000,
-       .ts_mode = 0,
+static const struct rtl2830_platform_data rtl2830_mxl5005s_platform_data = {
+       .clk = 28800000,
        .spec_inv = 0,
        .vtop = 0x3f,
        .krf = 0x04,
 {
        struct dvb_usb_device *d = adap_to_d(adap);
        struct rtl28xxu_priv *priv = d_to_priv(d);
-       const struct rtl2830_config *rtl2830_config;
+       struct rtl2830_platform_data *pdata = &priv->rtl2830_platform_data;
+       struct i2c_board_info board_info;
+       struct i2c_client *client;
        int ret;
 
        dev_dbg(&d->udev->dev, "%s:\n", __func__);
 
        switch (priv->tuner) {
        case TUNER_RTL2830_QT1010:
-               rtl2830_config = &rtl28xxu_rtl2830_qt1010_config;
+               *pdata = rtl2830_qt1010_platform_data;
                break;
        case TUNER_RTL2830_MT2060:
-               rtl2830_config = &rtl28xxu_rtl2830_mt2060_config;
+               *pdata = rtl2830_mt2060_platform_data;
                break;
        case TUNER_RTL2830_MXL5005S:
-               rtl2830_config = &rtl28xxu_rtl2830_mxl5005s_config;
+               *pdata = rtl2830_mxl5005s_platform_data;
                break;
        default:
                dev_err(&d->udev->dev, "%s: unknown tuner=%s\n",
        }
 
        /* attach demodulator */
-       adap->fe[0] = dvb_attach(rtl2830_attach, rtl2830_config, &d->i2c_adap);
-       if (!adap->fe[0]) {
+       memset(&board_info, 0, sizeof(board_info));
+       strlcpy(board_info.type, "rtl2830", I2C_NAME_SIZE);
+       board_info.addr = 0x10;
+       board_info.platform_data = pdata;
+       request_module("%s", board_info.type);
+       client = i2c_new_device(&d->i2c_adap, &board_info);
+       if (client == NULL || client->dev.driver == NULL) {
                ret = -ENODEV;
                goto err;
        }
 
+       if (!try_module_get(client->dev.driver->owner)) {
+               i2c_unregister_device(client);
+               ret = -ENODEV;
+               goto err;
+       }
+
+       adap->fe[0] = pdata->get_dvb_frontend(client);
+       priv->demod_i2c_adapter = pdata->get_i2c_adapter(client);
+
+       priv->i2c_client_demod = client;
+
        return 0;
 err:
        dev_dbg(&d->udev->dev, "%s: failed=%d\n", __func__, ret);
        int ret;
        struct dvb_usb_device *d = adap_to_d(adap);
        struct rtl28xxu_priv *priv = d_to_priv(d);
-       struct i2c_adapter *rtl2830_tuner_i2c;
        struct dvb_frontend *fe;
 
        dev_dbg(&d->udev->dev, "%s:\n", __func__);
 
-       /* use rtl2830 driver I2C adapter, for more info see rtl2830 driver */
-       rtl2830_tuner_i2c = rtl2830_get_tuner_i2c_adapter(adap->fe[0]);
-
        switch (priv->tuner) {
        case TUNER_RTL2830_QT1010:
                fe = dvb_attach(qt1010_attach, adap->fe[0],
-                               rtl2830_tuner_i2c, &rtl28xxu_qt1010_config);
+                               priv->demod_i2c_adapter,
+                               &rtl28xxu_qt1010_config);
                break;
        case TUNER_RTL2830_MT2060:
                fe = dvb_attach(mt2060_attach, adap->fe[0],
-                               rtl2830_tuner_i2c, &rtl28xxu_mt2060_config,
-                               1220);
+                               priv->demod_i2c_adapter,
+                               &rtl28xxu_mt2060_config, 1220);
                break;
        case TUNER_RTL2830_MXL5005S:
                fe = dvb_attach(mxl5005s_attach, adap->fe[0],
-                               rtl2830_tuner_i2c, &rtl28xxu_mxl5005s_config);
+                               priv->demod_i2c_adapter,
+                               &rtl28xxu_mxl5005s_config);
                break;
        default:
                fe = NULL;
        .i2c_algo = &rtl28xxu_i2c_algo,
        .read_config = rtl2831u_read_config,
        .frontend_attach = rtl2831u_frontend_attach,
+       .frontend_detach = rtl2832u_frontend_detach,
        .tuner_attach = rtl2831u_tuner_attach,
        .init = rtl28xxu_init,
        .get_rc_config = rtl2831u_get_rc_config,