upstream: [media] rtl2832: provide muxed I2C adapter
authorAntti Palosaari <crope@iki.fi>
Tue, 26 Nov 2013 15:53:46 +0000 (12:53 -0300)
committerChanho Park <chanho61.park@samsung.com>
Tue, 18 Nov 2014 02:54:49 +0000 (11:54 +0900)
RTL2832 provides gated / repeater I2C adapter for tuner.
Implement it as a muxed I2C adapter.

Signed-off-by: Antti Palosaari <crope@iki.fi>
Signed-off-by: Mauro Carvalho Chehab <m.chehab@samsung.com>
drivers/media/dvb-frontends/Kconfig
drivers/media/dvb-frontends/rtl2832.c
drivers/media/dvb-frontends/rtl2832.h
drivers/media/dvb-frontends/rtl2832_priv.h

index 4433071..73069ee 100644 (file)
@@ -441,7 +441,7 @@ config DVB_RTL2830
 
 config DVB_RTL2832
        tristate "Realtek RTL2832 DVB-T"
-       depends on DVB_CORE && I2C
+       depends on DVB_CORE && I2C && I2C_MUX
        default m if !MEDIA_SUBDRV_AUTOSELECT
        help
          Say Y when you want to support this frontend.
index ec668a7..b7b12e5 100644 (file)
@@ -897,9 +897,29 @@ static void rtl2832_release(struct dvb_frontend *fe)
        struct rtl2832_priv *priv = fe->demodulator_priv;
 
        dev_dbg(&priv->i2c->dev, "%s:\n", __func__);
+       i2c_del_mux_adapter(priv->i2c_adapter);
        kfree(priv);
 }
 
+static int rtl2832_select(struct i2c_adapter *adap, void *mux_priv, u32 chan)
+{
+       struct rtl2832_priv *priv = mux_priv;
+       return rtl2832_i2c_gate_ctrl(&priv->fe, 1);
+}
+
+static int rtl2832_deselect(struct i2c_adapter *adap, void *mux_priv, u32 chan)
+{
+       struct rtl2832_priv *priv = mux_priv;
+       return rtl2832_i2c_gate_ctrl(&priv->fe, 0);
+}
+
+struct i2c_adapter *rtl2832_get_i2c_adapter(struct dvb_frontend *fe)
+{
+       struct rtl2832_priv *priv = fe->demodulator_priv;
+       return priv->i2c_adapter;
+}
+EXPORT_SYMBOL(rtl2832_get_i2c_adapter);
+
 struct dvb_frontend *rtl2832_attach(const struct rtl2832_config *cfg,
        struct i2c_adapter *i2c)
 {
@@ -924,6 +944,12 @@ struct dvb_frontend *rtl2832_attach(const struct rtl2832_config *cfg,
        if (ret)
                goto err;
 
+       /* create muxed i2c adapter */
+       priv->i2c_adapter = i2c_add_mux_adapter(i2c, &i2c->dev, priv, 0, 0, 0,
+                       rtl2832_select, rtl2832_deselect);
+       if (priv->i2c_adapter == NULL)
+               goto err;
+
        /* create dvb_frontend */
        memcpy(&priv->fe.ops, &rtl2832_ops, sizeof(struct dvb_frontend_ops));
        priv->fe.demodulator_priv = priv;
index 58cdde0..bfe9249 100644 (file)
@@ -62,7 +62,13 @@ struct dvb_frontend *rtl2832_attach(
        const struct rtl2832_config *cfg,
        struct i2c_adapter *i2c
 );
+
+extern struct i2c_adapter *rtl2832_get_i2c_adapter(
+       struct dvb_frontend *fe
+);
+
 #else
+
 static inline struct dvb_frontend *rtl2832_attach(
        const struct rtl2832_config *config,
        struct i2c_adapter *i2c
@@ -71,6 +77,13 @@ static inline struct dvb_frontend *rtl2832_attach(
        pr_warn("%s: driver disabled by Kconfig\n", __func__);
        return NULL;
 }
+
+static inline struct i2c_adapter *rtl2832_get_i2c_adapter(
+       struct dvb_frontend *fe
+)
+{
+       return NULL;
+}
 #endif
 
 
index 4c845af..ec26c92 100644 (file)
 
 #include "dvb_frontend.h"
 #include "rtl2832.h"
+#include <linux/i2c-mux.h>
 
 struct rtl2832_priv {
        struct i2c_adapter *i2c;
+       struct i2c_adapter *i2c_adapter;
        struct dvb_frontend fe;
        struct rtl2832_config cfg;