Re: [PATCH v4 2/5] MFD: WL1273 FM Radio: MFD driver for the FM radio.

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



--- On Wed, 7/7/10, Pavan Savoy <pavan_savoy@xxxxxx> wrote:

> From: Pavan Savoy <pavan_savoy@xxxxxx>
> Subject: Re: [PATCH v4 2/5] MFD: WL1273 FM Radio: MFD driver for the FM radio.
> To: "Matti J. Aaltonen" <matti.j.aaltonen@xxxxxxxxx>, "Mauro Carvalho Chehab" <mchehab@xxxxxxxxxx>
> Cc: linux-media@xxxxxxxxxxxxxxx, hverkuil@xxxxxxxxx, eduardo.valentin@xxxxxxxxx, "pavan savoy" <pavan_savoy@xxxxxxxxxxx>
> Date: Wednesday, 7 July, 2010, 10:51 AM
> 
> --- On Tue, 6/7/10, Mauro Carvalho Chehab <mchehab@xxxxxxxxxx>
> wrote:
> 
> > From: Mauro Carvalho Chehab <mchehab@xxxxxxxxxx>
> > Subject: Re: [PATCH v4 2/5] MFD: WL1273 FM Radio: MFD
> driver for the FM radio.
> > To: "Matti J. Aaltonen" <matti.j.aaltonen@xxxxxxxxx>
> > Cc: linux-media@xxxxxxxxxxxxxxx,
> hverkuil@xxxxxxxxx,
> eduardo.valentin@xxxxxxxxx
> > Date: Tuesday, 6 July, 2010, 8:04 AM
> > Em 04-06-2010 07:34, Matti J.
> > Aaltonen escreveu:
> > > This is a parent driver for two child drivers:
> the
> > V4L2 driver and
> > > the ALSA codec driver. The MFD part provides the
> I2C
> > communication
> > > to the device and a couple of functions that are
> > called from both
> > > children.
> 
> Where can I have a look at the whole code ? As in some
> local tree, where all codes are put up ?
> Basically, we have a V4L2 driver for WL128x (using TTY as
> transport) and plan to push them soon ...
> 
> This would be a nice input ...

Anywhere I can have a look at the whole code?
We are trying to push out V4L2 drivers for FM on 128x, and intend to use several V4L2 new ioctls which you have defined (_BAND_) being one of them.

Also we plan to push FM Tx of it too.. TX_CLASS from V4L2 has most of the ioctls and plan to introduce the rest.

> 
> > > Signed-off-by: Matti J. Aaltonen <matti.j.aaltonen@xxxxxxxxx>
> > > ---
> > >  drivers/mfd/Kconfig       
> >      |    6 +
> > >  drivers/mfd/Makefile       
> >     |    2 +
> > >  drivers/mfd/wl1273-core.c   
> >    |  616
> > +++++++++++++++++++++++++++++++++++++++
> > >  include/linux/mfd/wl1273-core.h |  326
> > +++++++++++++++++++++
> > >  4 files changed, 950 insertions(+), 0
> > deletions(-)
> > >  create mode 100644 drivers/mfd/wl1273-core.c
> > >  create mode 100644
> > include/linux/mfd/wl1273-core.h
> > > 
> > > diff --git a/drivers/mfd/Kconfig
> > b/drivers/mfd/Kconfig
> > > index 413576a..5998a94 100644
> > > --- a/drivers/mfd/Kconfig
> > > +++ b/drivers/mfd/Kconfig
> > > @@ -135,6 +135,12 @@ config TWL4030_CODEC
> > >      select MFD_CORE
> > >      default n
> > >  
> > > +config WL1273_CORE
> > > +    bool
> > > +    depends on I2C
> > > +    select MFD_CORE
> > > +    default n
> > > +
> > >  config MFD_TMIO
> > >      bool
> > >      default n
> > > diff --git a/drivers/mfd/Makefile
> > b/drivers/mfd/Makefile
> > > index 78295d6..46e611d 100644
> > > --- a/drivers/mfd/Makefile
> > > +++ b/drivers/mfd/Makefile
> > > @@ -30,6 +30,8 @@
> > obj-$(CONFIG_TWL4030_CORE)    += twl-core.o
> > twl4030-irq.o twl6030-irq.o
> > >  obj-$(CONFIG_TWL4030_POWER)    +=
> > twl4030-power.o
> > >  obj-$(CONFIG_TWL4030_CODEC)    +=
> > twl4030-codec.o
> > >  
> > > +obj-$(CONFIG_WL1273_CORE)    +=
> > wl1273-core.o
> > > +
> > >  obj-$(CONFIG_MFD_MC13783)    +=
> > mc13783-core.o
> > >  
> > >  obj-$(CONFIG_MFD_CORE)   
> >     += mfd-core.o
> > > diff --git a/drivers/mfd/wl1273-core.c
> > b/drivers/mfd/wl1273-core.c
> > > new file mode 100644
> > > index 0000000..6c7dbba
> > > --- /dev/null
> > > +++ b/drivers/mfd/wl1273-core.c
> > > @@ -0,0 +1,616 @@
> > > +/*
> > > + * MFD driver for wl1273 FM radio and audio
> codec
> > submodules.
> > > + *
> > > + * Author:    Matti Aaltonen <matti.j.aaltonen@xxxxxxxxx>
> > > + *
> > > + * Copyright:   (C) 2010 Nokia
> > Corporation
> > > + *
> > > + * This program is free software; you can
> > redistribute it and/or modify
> > > + * it under the terms of the GNU General Public
> > License version 2 as
> > > + * published by the Free Software Foundation.
> > > + *
> > > + * 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.
> > > + *
> > > + * You should have received a copy of the GNU
> General
> > Public License
> > > + * along with this program; if not, write to the
> Free
> > Software
> > > + * Foundation, Inc., 51 Franklin St, Fifth
> Floor,
> > Boston, MA
> > > + * 02110-1301 USA
> > > + *
> > > + */
> > > +
> > > +#undef DEBUG
> > > +
> > > +#include <asm/unaligned.h>
> > > +#include <linux/completion.h>
> > > +#include <linux/delay.h>
> > > +#include <linux/i2c.h>
> > > +#include <linux/interrupt.h>
> > > +#include <linux/module.h>
> > > +#include <linux/types.h>
> > > +#include <linux/kernel.h>
> > > +#include <linux/fs.h>
> > > +#include <linux/platform_device.h>
> > > +#include <linux/mfd/core.h>
> > > +#include <linux/mfd/wl1273-core.h>
> > > +#include <media/v4l2-common.h>
> > > +
> > > +#define DRIVER_DESC "WL1273 FM Radio Core"
> > > +
> > > +#define
> > WL1273_IRQ_MASK     (WL1273_FR_EVENT   
> >     |    \
> > > +       
> >       WL1273_POW_ENB_EVENT)
> > > +
> > > +static const struct band_info bands[] = {
> > > +    /* USA & Europe */
> > > +    {
> > > +       
> > .bottom_frequency    = 87500,
> > > +       
> > .top_frequency        =
> > 108000,
> > > +       
> > .band       
> >     = V4L2_FM_BAND_OTHER,
> > > +    },
> > > +    /* Japan */
> > > +    {
> > > +       
> > .bottom_frequency    = 76000,
> > > +       
> > .top_frequency        =
> > 90000,
> > > +       
> > .band       
> >     = V4L2_FM_BAND_JAPAN,
> > > +    },
> > > +};
> > > +
> > > +/*
> > > + * static unsigned char radio_band - Band
> > > + *
> > > + * The bands are 0=Japan, 1=USA-Europe.
> USA-Europe is
> > the default.
> > > + */
> > > +static unsigned char radio_band = 1;
> > > +module_param(radio_band, byte, 0);
> > > +MODULE_PARM_DESC(radio_band, "Band: 0=Japan,
> > 1=USA-Europe*");
> > > +
> > > +/*
> > > + * static unsigned int rds_buf - the number of
> RDS
> > buffer blocks used.
> > > + *
> > > + * The default number is 100.
> > > + */
> > > +static unsigned int rds_buf = 100;
> > > +module_param(rds_buf, uint, 0);
> > > +MODULE_PARM_DESC(rds_buf, "RDS buffer entries:
> > *100*");
> > > +
> > > +int wl1273_fm_read_reg(struct wl1273_core *core,
> u8
> > reg, u16 *value)
> > > +{
> > > +    struct i2c_client *client =
> > core->i2c_dev;
> > > +    u8 b[2];
> > > +    int r;
> > > +
> > > +    r =
> > i2c_smbus_read_i2c_block_data(client, reg, 2, b);
> > > +    if (r != 2) {
> > > +       
> > dev_err(&client->dev, "%s: Read: %d fails.\n",
> > __func__, reg);
> > > +        return
> > -EREMOTEIO;
> > > +    }
> > > +
> > > +    *value = (u16)b[0] << 8 |
> > b[1];
> > > +
> > > +    return 0;
> > > +}
> > > +EXPORT_SYMBOL(wl1273_fm_read_reg);
> > > +
> > > +int wl1273_fm_write_cmd(struct wl1273_core
> *core, u8
> > cmd, u16 param)
> > > +{
> > > +    struct i2c_client *client =
> > core->i2c_dev;
> > > +    u8 buf[] = { (param >> 8)
> > & 0xff, param & 0xff };
> > > +    int r;
> > > +
> > > +    r =
> > i2c_smbus_write_i2c_block_data(client, cmd, 2, buf);
> > > +    if (r) {
> > > +       
> > dev_err(&client->dev, "%s: Cmd: %d fails.\n",
> > __func__, cmd);
> > > +        return r;
> > > +    }
> > > +
> > > +    return 0;
> > > +}
> > > +EXPORT_SYMBOL(wl1273_fm_write_cmd);
> > > +
> > > +int wl1273_fm_write_data(struct wl1273_core
> *core, u8
> > *data, u16 len)
> > > +{
> > > +    struct i2c_client *client =
> > core->i2c_dev;
> > > +    struct i2c_msg msg[1];
> > > +    int r;
> > > +
> > > +    msg[0].addr = client->addr;
> > > +    msg[0].flags = 0;
> > > +    msg[0].buf = data;
> > > +    msg[0].len = len;
> > > +
> > > +    r =
> > i2c_transfer(client->adapter, msg, 1);
> > > +
> > > +    if (r != 1) {
> > > +       
> > dev_err(&client->dev, "%s: write error.\n",
> > __func__);
> > > +        return
> > -EREMOTEIO;
> > > +    }
> > > +
> > > +    return 0;
> > > +}
> > > +EXPORT_SYMBOL(wl1273_fm_write_data);
> > > +
> > > +/**
> > > + * wl1273_fm_set_audio() -    Set
> > audio mode.
> > > + * @core:       
> >     A pointer to the device struct.
> > > + * @new_mode:       
> >     The new audio mode.
> > > + *
> > > + * Audio modes are WL1273_AUDIO_DIGITAL and
> > WL1273_AUDIO_ANALOG.
> > > + */
> > > +int wl1273_fm_set_audio(struct wl1273_core
> *core,
> > unsigned int new_mode)
> > > +{
> > > +    int r = 0;
> > > +
> > > +    if (core->mode ==
> > WL1273_MODE_OFF ||
> > > +        core->mode ==
> > WL1273_MODE_SUSPENDED)
> > > +        return -EPERM;
> > > +
> > > +    if (core->mode ==
> > WL1273_MODE_RX && new_mode ==
> WL1273_AUDIO_DIGITAL)
> > {
> > > +        r =
> > wl1273_fm_write_cmd(core, WL1273_PCM_MODE_SET,
> > > +       
> >            
> > WL1273_PCM_DEF_MODE);
> > > +        if (r)
> > > +       
> >     goto out;
> > > +
> > > +        r =
> > wl1273_fm_write_cmd(core, WL1273_I2S_MODE_CONFIG_SET,
> > > +       
> >            
> > core->i2s_mode);
> > > +        if (r)
> > > +       
> >     goto out;
> > > +
> > > +        r =
> > wl1273_fm_write_cmd(core, WL1273_AUDIO_ENABLE,
> > > +       
> >            
> > WL1273_AUDIO_ENABLE_I2S);
> > > +        if (r)
> > > +       
> >     goto out;
> > > +
> > > +    } else if (core->mode ==
> > WL1273_MODE_RX &&
> > > +       
> >    new_mode == WL1273_AUDIO_ANALOG) {
> > > +        r =
> > wl1273_fm_write_cmd(core, WL1273_AUDIO_ENABLE,
> > > +       
> >            
> > WL1273_AUDIO_ENABLE_ANALOG);
> > > +        if (r)
> > > +       
> >     goto out;
> > > +
> > > +    } else if (core->mode ==
> > WL1273_MODE_TX &&
> > > +       
> >    new_mode == WL1273_AUDIO_DIGITAL) {
> > > +        r =
> > wl1273_fm_write_cmd(core, WL1273_I2S_MODE_CONFIG_SET,
> > > +       
> >            
> > core->i2s_mode);
> > > +        if (r)
> > > +       
> >     goto out;
> > > +
> > > +        r =
> > wl1273_fm_write_cmd(core, WL1273_AUDIO_IO_SET,
> > > +       
> >            
> > WL1273_AUDIO_IO_SET_I2S);
> > > +        if (r)
> > > +       
> >     goto out;
> > > +
> > > +    } else if (core->mode ==
> > WL1273_MODE_TX &&
> > > +       
> >    new_mode == WL1273_AUDIO_ANALOG) {
> > > +        r =
> > wl1273_fm_write_cmd(core, WL1273_AUDIO_IO_SET,
> > > +       
> >            
> > WL1273_AUDIO_IO_SET_ANALOG);
> > > +        if (r)
> > > +       
> >     goto out;
> > > +    }
> > > +
> > > +    core->audio_mode = new_mode;
> > > +
> > > +out:
> > > +    return r;
> > > +}
> > > +EXPORT_SYMBOL(wl1273_fm_set_audio);
> > > +
> > > +/**
> > > + * wl1273_fm_set_volume() -    Set
> > volume.
> > > + * @core:       
> >     A pointer to the device struct.
> > > + * @volume:       
> >     The new volume value.
> > > + */
> > > +int wl1273_fm_set_volume(struct wl1273_core
> *core,
> > unsigned int volume)
> > > +{
> > > +    u16 val;
> > > +    int r;
> > > +
> > > +    if (volume >
> > WL1273_MAX_VOLUME)
> > > +        return
> > -EINVAL;
> > > +
> > > +    if (core->volume == volume)
> > > +        return 0;
> > > +
> > > +    val = volume;
> > > +    r = wl1273_fm_read_reg(core,
> > WL1273_VOLUME_SET, &val);
> > > +    if (r)
> > > +        return r;
> > > +
> > > +    core->volume = volume;
> > > +    return 0;
> > > +}
> > > +EXPORT_SYMBOL(wl1273_fm_set_volume);
> > > +
> > > +#define
> > WL1273_RDS_FIFO_EMPTY(status)    (!(1
> > << 6 &  status))
> > > +
> > > +#define
> > WL1273_RDS_CORRECTABLE_ERROR    (1 <<
> > 3)
> > > +#define
> > WL1273_RDS_UNCORRECTABLE_ERROR    (1 <<
> > 4)
> > > +
> > > +static int wl1273_fm_rds(struct wl1273_core
> *core)
> > > +{
> > > +    struct i2c_client *client =
> > core->i2c_dev;
> > > +    struct device *dev =
> > &client->dev;
> > > +    struct v4l2_rds_data rds = { 0, 0,
> > 0 };
> > > +    struct i2c_msg msg[] = {
> > > +        {
> > > +       
> >     .addr = client->addr,
> > > +       
> >     .flags = 0,
> > > +       
> >     .buf = b0,
> > > +       
> >     .len = 1
> > > +        },
> > > +        {
> > > +       
> >     .addr = client->addr,
> > > +       
> >     .flags = I2C_M_RD,
> > > +       
> >     .buf = (u8 *) &rds,
> > > +       
> >     .len = 3
> > > +        }
> > > +    };
> > > +    u8 b0[] = { WL1273_RDS_DATA_GET },
> > status;
> > > +    u16 val;
> > > +    int r;
> > > +
> > > +    r = wl1273_fm_read_reg(core,
> > WL1273_RDS_SYNC_GET, &val);
> > > +    if (r)
> > > +        return r;
> > > +
> > > +    /* Is RDS decoder synchronized?
> > */
> > > +    if ((val & 0x01) == 0)
> > > +        return
> > -EAGAIN;
> > > +
> > > +    /* copy all four RDS blocks to
> > internal buffer */
> > > +    do {
> > > +        r =
> > i2c_transfer(client->adapter, msg, 2);
> > > +        if (r != 2) {
> > > +       
> >     dev_err(dev, WL1273_FM_DRIVER_NAME
> > > +       
> >         ": %s: read_rds error
> > r == %i)\n",
> > > +       
> >         __func__, r);
> > > +        }
> > > +
> > > +        status =
> > rds.block;
> > > +
> > > +        if
> > (WL1273_RDS_FIFO_EMPTY(status))
> > > +       
> >     break;
> > > +
> > > +        /* copy RDS
> > block to internal buffer */
> > > +       
> > memcpy(&core->buffer[core->wr_index],
> &rds,
> > 3);
> > > +       
> > core->wr_index += 3;
> > > +
> > > +        /* copy the
> > error bits to standard positions */
> > > +        if
> > (WL1273_RDS_UNCORRECTABLE_ERROR & status) {
> > > +       
> >     rds.block |= V4L2_RDS_BLOCK_ERROR;
> > > +       
> >     rds.block &=
> > ~V4L2_RDS_BLOCK_CORRECTED;
> > > +        } else if 
> > (WL1273_RDS_CORRECTABLE_ERROR & status) {
> > > +       
> >     rds.block &= ~V4L2_RDS_BLOCK_ERROR;
> > > +       
> >     rds.block |= V4L2_RDS_BLOCK_CORRECTED;
> > > +        } else {
> > > +       
> >     rds.block &= ~V4L2_RDS_BLOCK_ERROR;
> > > +       
> >     rds.block &=
> > ~V4L2_RDS_BLOCK_CORRECTED;
> > > +        }
> > > +
> > > +        /* wrap write
> > pointer */
> > > +        if
> > (core->wr_index >= core->buf_size)
> > > +       
> >     core->wr_index = 0;
> > > +
> > > +        /* check for
> > overflow & start over */
> > > +        if
> > (core->wr_index == core->rd_index) {
> > > +       
> >     dev_dbg(dev, "RDS OVERFLOW");
> > > +
> > > +       
> >     core->rd_index = 0;
> > > +       
> >     core->wr_index = 0;
> > > +       
> >     break;
> > > +        }
> > > +    } while
> > (!WL1273_RDS_FIFO_EMPTY(status));
> > > +
> > > +    /* wake up read queue */
> > > +    if (core->wr_index !=
> > core->rd_index)
> > > +       
> > wake_up_interruptible(&core->read_queue);
> > > +
> > > +    return 0;
> > > +}
> > > +
> > > +static void wl1273_fm_rds_work(struct
> wl1273_core
> > *core)
> > > +{
> > > +    wl1273_fm_rds(core);
> > > +}
> > > +
> > > +static irqreturn_t
> wl1273_fm_irq_thread_handler(int
> > irq, void *dev_id)
> > > +{
> > > +    int r;
> > > +    u16 flags;
> > > +    struct wl1273_core *core =
> > dev_id;
> > > +
> > > +    r = wl1273_fm_read_reg(core,
> > WL1273_FLAG_GET, &flags);
> > > +    if (r)
> > > +        goto out;
> > > +
> > > +    if (flags & WL1273_BL_EVENT)
> > {
> > > +       
> > core->irq_received = flags;
> > > +       
> > dev_dbg(&core->i2c_dev->dev, "IRQ: BL\n");
> > > +    }
> > > +
> > > +    if (flags & WL1273_RDS_EVENT)
> > {
> > > +        msleep(200);
> > > +
> > > +       
> > wl1273_fm_rds_work(core);
> > > +    }
> > > +
> > > +    if (flags &
> > WL1273_BBLK_EVENT)
> > > +       
> > dev_dbg(&core->i2c_dev->dev, "IRQ:
> BBLK\n");
> > > +
> > > +    if (flags &
> > WL1273_LSYNC_EVENT)
> > > +       
> > dev_dbg(&core->i2c_dev->dev, "IRQ:
> LSYNC\n");
> > > +
> > > +    if (flags & WL1273_LEV_EVENT)
> > {
> > > +        u16 level;
> > > +
> > > +        r =
> > wl1273_fm_read_reg(core, WL1273_RSSI_LVL_GET,
> &level);
> > > +
> > > +        if (r)
> > > +       
> >     goto out;
> > > +
> > > +        if (level >
> > 14)
> > > +       
> >     dev_dbg(&core->i2c_dev->dev,
> > "IRQ: LEV: 0x%x04\n",
> > > +       
> >         level);
> > > +    }
> > > +
> > > +    if (flags &
> > WL1273_IFFR_EVENT)
> > > +       
> > dev_dbg(&core->i2c_dev->dev, "IRQ:
> IFFR\n");
> > > +
> > > +    if (flags & WL1273_PI_EVENT)
> > > +       
> > dev_dbg(&core->i2c_dev->dev, "IRQ: PI\n");
> > > +
> > > +    if (flags & WL1273_PD_EVENT)
> > > +       
> > dev_dbg(&core->i2c_dev->dev, "IRQ: PD\n");
> > > +
> > > +    if (flags &
> > WL1273_STIC_EVENT)
> > > +       
> > dev_dbg(&core->i2c_dev->dev, "IRQ:
> STIC\n");
> > > +
> > > +    if (flags & WL1273_MAL_EVENT)
> > > +       
> > dev_dbg(&core->i2c_dev->dev, "IRQ: MAL\n");
> > > +
> > > +    if (flags &
> > WL1273_POW_ENB_EVENT) {
> > > +       
> > complete(&core->busy);
> > > +       
> > dev_dbg(&core->i2c_dev->dev, "NOT BUSY\n");
> > > +       
> > dev_dbg(&core->i2c_dev->dev, "IRQ:
> POW_ENB\n");
> > > +    }
> > > +
> > > +    if (flags &
> > WL1273_SCAN_OVER_EVENT)
> > > +       
> > dev_dbg(&core->i2c_dev->dev, "IRQ:
> SCAN_OVER\n");
> > > +
> > > +    if (flags &
> > WL1273_ERROR_EVENT)
> > > +       
> > dev_dbg(&core->i2c_dev->dev, "IRQ:
> ERROR\n");
> > > +
> > > +    if (flags & WL1273_FR_EVENT)
> > {
> > > +        u16 freq;
> > > +
> > > +       
> > dev_dbg(&core->i2c_dev->dev, "IRQ: FR:\n");
> > > +
> > > +        if
> > (core->mode == WL1273_MODE_RX) {
> > > +       
> >     r = wl1273_fm_write_cmd(core,
> > WL1273_TUNER_MODE_SET,
> > > +       
> >            
> >     TUNER_MODE_STOP_SEARCH);
> > > +       
> >     if (r) {
> > > +       
> >        
> > dev_err(&core->i2c_dev->dev,
> > > +       
> >            
> > "%s: TUNER_MODE_SET fails: %d\n",
> > > +       
> >            
> > __func__, r);
> > > +       
> >         goto out;
> > > +       
> >     }
> > > +
> > > +       
> >     r = wl1273_fm_read_reg(core,
> > WL1273_FREQ_SET, &freq);
> > > +       
> >     if (r)
> > > +       
> >         goto out;
> > > +
> > > +       
> >     core->rx_frequency =
> > > +       
> >        
> > bands[core->band].bottom_frequency +
> > > +       
> >         freq * 50;
> > > +
> > > +       
> >     /*
> > > +       
> >      *  The driver works
> > better with this msleep,
> > > +       
> >      *  the documentation
> > doesn't mention it.
> > > +       
> >      */
> > > +       
> >     msleep(10);
> > 
> > 
> > msleep on an irq handler? You shouldn't be doing it!
> You're
> > not allowed to sleep
> > during IRQ time. Kernel can panic here. You'll
> probably
> > need to defer work and
> > handle it outside irq time.
> > > +
> > > +       
> >     dev_dbg(&core->i2c_dev->dev,
> > "%dkHz\n",
> > > +       
> >        
> > core->rx_frequency);
> > > +
> > > +        } else {
> > > +       
> >     r = wl1273_fm_read_reg(core,
> > WL1273_CHANL_SET, &freq);
> > > +       
> >     if (r)
> > > +       
> >         goto out;
> > > +
> > > +       
> >     dev_dbg(&core->i2c_dev->dev,
> > "%dkHz\n", freq);
> > > +        }
> > > +       
> > dev_dbg(&core->i2c_dev->dev, "%s: NOT
> BUSY\n",
> > __func__);
> > > +    }
> > > +
> > > +out:
> > > +    wl1273_fm_write_cmd(core,
> > WL1273_INT_MASK_SET,
> > > +       
> >         core->irq_flags);
> > > +    complete(&core->busy);
> > > +
> > > +    return IRQ_HANDLED;
> > > +}
> > > +
> > > +static struct i2c_device_id
> wl1273_driver_id_table[]
> > = {
> > > +    { WL1273_FM_DRIVER_NAME, 0 },
> > > +    { }
> > > +};
> > > +MODULE_DEVICE_TABLE(i2c,
> wl1273_driver_id_table);
> > > +
> > > +static int wl1273_core_remove(struct i2c_client
> > *client)
> > > +{
> > > +    struct wl1273_core *core =
> > i2c_get_clientdata(client);
> > > +    struct wl1273_fm_platform_data
> > *pdata =
> > > +       
> > client->dev.platform_data;
> > > +
> > > +    dev_dbg(&client->dev,
> > "%s\n", __func__);
> > > +
> > > +   
> > mfd_remove_devices(&client->dev);
> > > +    i2c_set_clientdata(client, core);
> > > +
> > > +    free_irq(client->irq, core);
> > > +    pdata->free_resources();
> > > +
> > > +    kfree(core->buffer);
> > > +    kfree(core);
> > > +
> > > +    return 0;
> > > +}
> > > +
> > > +static int __devinit wl1273_core_probe(struct
> > i2c_client *client,
> > > +       
> >            
> >    const struct i2c_device_id *id)
> > > +{
> > > +    struct wl1273_fm_platform_data
> > *pdata = client->dev.platform_data;
> > > +    int r = 0;
> > > +    struct wl1273_core *core;
> > > +    int children = 0;
> > > +
> > > +    dev_dbg(&client->dev,
> > "%s\n", __func__);
> > > +
> > > +    if (!pdata) {
> > > +       
> > dev_err(&client->dev, "No platform data.\n");
> > > +        return
> > -EINVAL;
> > > +    }
> > > +
> > > +    core = kzalloc(sizeof(*core),
> > GFP_KERNEL);
> > > +    if (!core)
> > > +        return
> > -ENOMEM;
> > > +
> > > +    /* RDS buffer allocation */
> > > +    core->buf_size = rds_buf * 3;
> > > +    core->buffer =
> > kmalloc(core->buf_size, GFP_KERNEL);
> > > +    if (!core->buffer) {
> > > +       
> > dev_err(&client->dev,
> > > +       
> >     "Cannot allocate memory for RDS
> > buffer.\n");
> > > +        r = -ENOMEM;
> > > +        goto
> > err_kmalloc;
> > > +    }
> > > +
> > > +    core->irq_flags =
> > WL1273_IRQ_MASK;
> > > +    core->i2c_dev = client;
> > > +    core->rds_on = false;
> > > +    core->mode = WL1273_MODE_OFF;
> > > +    core->tx_power = 4;
> > > +    core->audio_mode =
> > WL1273_AUDIO_ANALOG;
> > > +    core->band = radio_band;
> > > +    core->bands = bands;
> > > +    core->number_of_bands =
> > ARRAY_SIZE(bands);
> > > +    core->i2s_mode =
> > WL1273_I2S_DEF_MODE;
> > > +    core->channel_number = 2;
> > > +    core->volume =
> > WL1273_DEFAULT_VOLUME;
> > > +    core->rx_frequency =
> > bands[core->band].bottom_frequency;
> > > +    core->tx_frequency =
> > bands[core->band].top_frequency;
> > > +
> > > +    dev_dbg(&client->dev,
> > "radio_band: %d\n", radio_band);
> > > +
> > > +    mutex_init(&core->lock);
> > > +
> > > +    pdata =
> > client->dev.platform_data;
> > > +    if (pdata) {
> > > +        r =
> > pdata->request_resources(client);
> > > +        if (r) {
> > > +       
> >     dev_err(&client->dev,
> > WL1273_FM_DRIVER_NAME
> > > +       
> >         ": Cannot get platform
> > data\n");
> > > +       
> >     goto err_new_mixer;
> > > +        }
> > > +
> > > +        r =
> > request_threaded_irq(client->irq, NULL,
> > > +       
> >        
> >      wl1273_fm_irq_thread_handler,
> > > +       
> >        
> >      IRQF_ONESHOT |
> > IRQF_TRIGGER_FALLING,
> > > +       
> >        
> >      "wl1273-fm", core);
> > > +        if (r < 0)
> > {
> > > +       
> >     dev_err(&client->dev,
> > WL1273_FM_DRIVER_NAME
> > > +       
> >         ": Unable to register
> > IRQ handler\n");
> > > +       
> >     goto err_request_irq;
> > > +        }
> > > +    } else {
> > > +       
> > dev_err(&client->dev, WL1273_FM_DRIVER_NAME ":
> Core
> > WL1273 IRQ"
> > > +       
> >     " not configured");
> > > +        r = -EINVAL;
> > > +        goto
> > err_new_mixer;
> > > +    }
> > > +
> > > +   
> > init_completion(&core->busy);
> > > +   
> > init_waitqueue_head(&core->read_queue);
> > > +
> > > +    i2c_set_clientdata(client, core);
> > > +
> > > +    if (pdata->children &
> > WL1273_RADIO_CHILD) {
> > > +        struct mfd_cell
> > *cell = &core->cells[children];
> > > +       
> > dev_dbg(&client->dev, "%s: Have V4L2.\n",
> __func__);
> > > +        cell->name =
> > "wl1273_fm_radio";
> > > +       
> > cell->platform_data = &core;
> > > +       
> > cell->data_size = sizeof(core);
> > > +        children++;
> > > +    }
> > > +
> > > +    if (pdata->children &
> > WL1273_CODEC_CHILD) {
> > > +        struct mfd_cell
> > *cell = &core->cells[children];
> > > +       
> > dev_dbg(&client->dev, "%s: Have codec.\n",
> > __func__);
> > > +        cell->name =
> > "wl1273_codec_audio";
> > > +       
> > cell->platform_data = &core;
> > > +       
> > cell->data_size = sizeof(core);
> > > +        children++;
> > > +    }
> > > +
> > > +    if (children) {
> > > +       
> > dev_dbg(&client->dev, "%s: Have children.\n",
> > __func__);
> > > +        r =
> > mfd_add_devices(&client->dev, -1,
> core->cells,
> > > +       
> >            
> > children, NULL, 0);
> > > +    } else {
> > > +       
> > dev_err(&client->dev, "No platform data found
> for
> > children.\n");
> > > +        r = -ENODEV;
> > > +    }
> > > +
> > > +    if (!r)
> > > +        return 0;
> > > +
> > > +    i2c_set_clientdata(client, NULL);
> > > +    kfree(core);
> > > +    free_irq(client->irq, core);
> > > +err_request_irq:
> > > +    pdata->free_resources();
> > > +err_new_mixer:
> > > +    kfree(core->buffer);
> > > +err_kmalloc:
> > > +    kfree(core);
> > > +    dev_dbg(&client->dev,
> > "%s\n", __func__);
> > > +
> > > +    return r;
> > > +}
> > > +
> > > +static struct i2c_driver wl1273_core_driver = {
> > > +    .driver = {
> > > +        .name =
> > WL1273_FM_DRIVER_NAME,
> > > +    },
> > > +    .probe = wl1273_core_probe,
> > > +    .id_table =
> > wl1273_driver_id_table,
> > > +    .remove =
> > __devexit_p(wl1273_core_remove),
> > > +};
> > > +
> > > +static int __init wl1273_core_init(void)
> > > +{
> > > +    int r;
> > > +
> > > +    r =
> > i2c_add_driver(&wl1273_core_driver);
> > > +    if (r) {
> > > +       
> > pr_err(WL1273_FM_DRIVER_NAME
> > > +           
> >    ": driver registration failed\n");
> > > +        return r;
> > > +    }
> > > +
> > > +    return 0;
> > > +}
> > > +
> > > +static void __exit wl1273_core_exit(void)
> > > +{
> > > +    flush_scheduled_work();
> > > +
> > > +   
> > i2c_del_driver(&wl1273_core_driver);
> > > +}
> > > +late_initcall(wl1273_core_init);
> > > +module_exit(wl1273_core_exit);
> > > +
> > > +MODULE_AUTHOR("Matti Aaltonen <matti.j.aaltonen@xxxxxxxxx>");
> > > +MODULE_DESCRIPTION(DRIVER_DESC);
> > > +MODULE_LICENSE("GPL");
> > > diff --git a/include/linux/mfd/wl1273-core.h
> > b/include/linux/mfd/wl1273-core.h
> > > new file mode 100644
> > > index 0000000..81c9743
> > > --- /dev/null
> > > +++ b/include/linux/mfd/wl1273-core.h
> > > @@ -0,0 +1,326 @@
> > > +/*
> > > + * include/media/radio/radio-wl1273.h
> > > + *
> > > + * Some definitions for the wl1273 radio
> > receiver/transmitter chip.
> > > + *
> > > + * Copyright (C) Nokia Corporation
> > > + * Author: Matti J. Aaltonen <matti.j.aaltonen@xxxxxxxxx>
> > > + *
> > > + * This program is free software; you can
> > redistribute it and/or
> > > + * modify it under the terms of the GNU General
> > Public License
> > > + * version 2 as published by the Free Software
> > Foundation.
> > > + *
> > > + * 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.
> > > + *
> > > + * You should have received a copy of the GNU
> General
> > Public License
> > > + * along with this program; if not, write to the
> Free
> > Software
> > > + * Foundation, Inc., 51 Franklin St, Fifth
> Floor,
> > Boston, MA
> > > + * 02110-1301 USA
> > > + */
> > > +
> > > +#ifndef RADIO_WL1273_H
> > > +#define RADIO_WL1273_H
> > > +
> > > +#include <linux/i2c.h>
> > > +#include <linux/mfd/core.h>
> > > +
> > > +#define WL1273_FM_DRIVER_NAME   
> > "wl1273-fm"
> > > +#define RX71_FM_I2C_ADDR    0x22
> > > +
> > > +#define WL1273_STEREO_GET   
> >     0
> > > +#define WL1273_RSSI_LVL_GET   
> >     1
> > > +#define WL1273_IF_COUNT_GET   
> >     2
> > > +#define WL1273_FLAG_GET   
> >         3
> > > +#define WL1273_RDS_SYNC_GET   
> >     4
> > > +#define WL1273_RDS_DATA_GET   
> >     5
> > > +#define WL1273_FREQ_SET   
> >         10
> > > +#define WL1273_AF_FREQ_SET   
> >     11
> > > +#define WL1273_MOST_MODE_SET   
> >     12
> > > +#define WL1273_MOST_BLEND_SET   
> >     13
> > > +#define WL1273_DEMPH_MODE_SET   
> >     14
> > > +#define WL1273_SEARCH_LVL_SET   
> >     15
> > > +#define WL1273_BAND_SET   
> >         16
> > > +#define WL1273_MUTE_STATUS_SET   
> >     17
> > > +#define WL1273_RDS_PAUSE_LVL_SET   
> > 18
> > > +#define WL1273_RDS_PAUSE_DUR_SET   
> > 19
> > > +#define WL1273_RDS_MEM_SET   
> >     20
> > > +#define WL1273_RDS_BLK_B_SET   
> >     21
> > > +#define WL1273_RDS_MSK_B_SET   
> >     22
> > > +#define WL1273_RDS_PI_MASK_SET   
> >     23
> > > +#define WL1273_RDS_PI_SET   
> >     24
> > > +#define WL1273_RDS_SYSTEM_SET   
> >     25
> > > +#define WL1273_INT_MASK_SET   
> >     26
> > > +#define WL1273_SEARCH_DIR_SET   
> >     27
> > > +#define WL1273_VOLUME_SET   
> >     28
> > > +#define WL1273_AUDIO_ENABLE   
> >     29
> > > +#define WL1273_PCM_MODE_SET   
> >     30
> > > +#define WL1273_I2S_MODE_CONFIG_SET   
> > 31
> > > +#define WL1273_POWER_SET   
> >     32
> > > +#define WL1273_INTX_CONFIG_SET   
> >     33
> > > +#define WL1273_PULL_EN_SET   
> >     34
> > > +#define WL1273_HILO_SET   
> >         35
> > > +#define WL1273_SWITCH2FREF   
> >     36
> > > +#define WL1273_FREQ_DRIFT_REPORT   
> > 37
> > > +
> > > +#define WL1273_PCE_GET   
> >         40
> > > +#define WL1273_FIRM_VER_GET   
> >     41
> > > +#define WL1273_ASIC_VER_GET   
> >     42
> > > +#define WL1273_ASIC_ID_GET   
> >     43
> > > +#define WL1273_MAN_ID_GET   
> >     44
> > > +#define WL1273_TUNER_MODE_SET   
> >     45
> > > +#define WL1273_STOP_SEARCH   
> >     46
> > > +#define WL1273_RDS_CNTRL_SET   
> >     47
> > > +
> > > +#define WL1273_WRITE_HARDWARE_REG   
> > 100
> > > +#define WL1273_CODE_DOWNLOAD   
> >     101
> > > +#define WL1273_RESET   
> >         102
> > > +
> > > +#define WL1273_FM_POWER_MODE   
> >     254
> > > +#define WL1273_FM_INTERRUPT   
> >     255
> > > +
> > > +/* Transmitter API */
> > > +
> > > +#define WL1273_CHANL_SET   
> >     55
> > > +#define WL1273_SCAN_SPACING_SET   
> >     56
> > > +#define WL1273_REF_SET   
> >         57
> > > +#define WL1273_POWER_ENB_SET   
> >     90
> > > +#define WL1273_POWER_ATT_SET   
> >     58
> > > +#define WL1273_POWER_LEV_SET   
> >     59
> > > +#define WL1273_AUDIO_DEV_SET   
> >     60
> > > +#define WL1273_PILOT_DEV_SET   
> >     61
> > > +#define WL1273_RDS_DEV_SET   
> >     62
> > > +#define WL1273_PUPD_SET   
> >         91
> > > +#define WL1273_AUDIO_IO_SET   
> >     63
> > > +#define WL1273_PREMPH_SET   
> >     64
> > > +#define WL1273_MONO_SET   
> >         66
> > > +#define WL1273_MUTE   
> >         92
> > > +#define WL1273_MPX_LMT_ENABLE   
> >     67
> > > +#define WL1273_PI_SET   
> >         93
> > > +#define WL1273_ECC_SET   
> >         69
> > > +#define WL1273_PTY   
> >         70
> > > +#define WL1273_AF   
> >         71
> > > +#define WL1273_DISPLAY_MODE   
> >     74
> > > +#define WL1273_RDS_REP_SET   
> >     77
> > > +#define WL1273_RDS_CONFIG_DATA_SET   
> > 98
> > > +#define WL1273_RDS_DATA_SET   
> >     99
> > > +#define WL1273_RDS_DATA_ENB   
> >     94
> > > +#define WL1273_TA_SET   
> >         78
> > > +#define WL1273_TP_SET   
> >         79
> > > +#define WL1273_DI_SET   
> >         80
> > > +#define WL1273_MS_SET   
> >         81
> > > +#define WL1273_PS_SCROLL_SPEED   
> >     82
> > > +#define WL1273_TX_AUDIO_LEVEL_TEST   
> > 96
> > > +#define
> > WL1273_TX_AUDIO_LEVEL_TEST_THRESHOLD    73
> > > +#define
> > WL1273_TX_AUDIO_INPUT_LEVEL_RANGE_SET    54
> > > +#define WL1273_RX_ANTENNA_SELECT   
> > 87
> > > +#define WL1273_I2C_DEV_ADDR_SET   
> >     86
> > > +#define
> > WL1273_REF_ERR_CALIB_PARAM_SET   
> >     88
> > > +#define
> > WL1273_REF_ERR_CALIB_PERIODICITY_SET    89
> > > +#define WL1273_SOC_INT_TRIGGER   
> >         52
> > > +#define WL1273_SOC_AUDIO_PATH_SET   
> >     83
> > > +#define WL1273_SOC_PCMI_OVERRIDE   
> >     84
> > > +#define WL1273_SOC_I2S_OVERRIDE   
> >     85
> > > +#define
> > WL1273_RSSI_BLOCK_SCAN_FREQ_SET    95
> > > +#define
> > WL1273_RSSI_BLOCK_SCAN_START    97
> > > +#define
> > WL1273_RSSI_BLOCK_SCAN_DATA_GET     5
> > > +#define
> > WL1273_READ_FMANT_TUNE_VALUE   
> >     104
> > > +
> > > +#define WL1273_RDS_OFF   
> >     0
> > > +#define WL1273_RDS_ON   
> >     1
> > > +#define WL1273_RDS_RESET    2
> > > +
> > > +#define WL1273_AUDIO_DIGITAL    0
> > > +#define WL1273_AUDIO_ANALOG    1
> > > +
> > > +#define WL1273_MODE_RX   
> >     0
> > > +#define WL1273_MODE_TX   
> >     1
> > > +#define WL1273_MODE_OFF   
> >     2
> > > +#define WL1273_MODE_SUSPENDED    3
> > > +
> > > +#define WL1273_RADIO_CHILD    (1
> > << 0)
> > > +#define WL1273_CODEC_CHILD    (1
> > << 1)
> > > +
> > > +#define WL1273_RX_MONO   
> >     1
> > > +#define WL1273_RX_STEREO    0
> > > +#define WL1273_TX_MONO   
> >     0
> > > +#define WL1273_TX_STEREO    1
> > > +
> > > +#define WL1273_MAX_VOLUME    0xffff
> > > +#define WL1273_DEFAULT_VOLUME   
> > 0x78b8
> > > +
> > > +/* I2S protocol, left channel first, data width
> 16
> > bits */
> > > +#define WL1273_PCM_DEF_MODE   
> >     0x00
> > > +
> > > +/* Rx */
> > > +#define WL1273_AUDIO_ENABLE_I2S   
> >     (1 << 0)
> > > +#define WL1273_AUDIO_ENABLE_ANALOG   
> > (1 << 1)
> > > +
> > > +/* Tx */
> > > +#define WL1273_AUDIO_IO_SET_ANALOG   
> > 0
> > > +#define WL1273_AUDIO_IO_SET_I2S   
> >     1
> > > +
> > > +#define WL1273_POWER_SET_OFF   
> >     0
> > > +#define WL1273_POWER_SET_FM   
> >     (1 << 0)
> > > +#define WL1273_POWER_SET_RDS   
> >     (1 << 1)
> > > +#define WL1273_POWER_SET_RETENTION   
> > (1 << 4)
> > > +
> > > +#define WL1273_PUPD_SET_OFF   
> >     0x00
> > > +#define WL1273_PUPD_SET_ON   
> >     0x01
> > > +#define WL1273_PUPD_SET_RETENTION   
> > 0x10
> > > +
> > > +/* I2S mode */
> > > +#define WL1273_IS2_WIDTH_32    0x0
> > > +#define WL1273_IS2_WIDTH_40    0x1
> > > +#define WL1273_IS2_WIDTH_22_23    0x2
> > > +#define WL1273_IS2_WIDTH_23_22    0x3
> > > +#define WL1273_IS2_WIDTH_48    0x4
> > > +#define WL1273_IS2_WIDTH_50    0x5
> > > +#define WL1273_IS2_WIDTH_60    0x6
> > > +#define WL1273_IS2_WIDTH_64    0x7
> > > +#define WL1273_IS2_WIDTH_80    0x8
> > > +#define WL1273_IS2_WIDTH_96    0x9
> > > +#define WL1273_IS2_WIDTH_128    0xa
> > > +#define WL1273_IS2_WIDTH    0xf
> > > +
> > > +#define WL1273_IS2_FORMAT_STD    (0x0
> > << 4)
> > > +#define WL1273_IS2_FORMAT_LEFT    (0x1
> > << 4)
> > > +#define WL1273_IS2_FORMAT_RIGHT   
> > (0x2 << 4)
> > > +#define WL1273_IS2_FORMAT_USER    (0x3
> > << 4)
> > > +
> > > +#define WL1273_IS2_MASTER    (0x0
> > << 6)
> > > +#define WL1273_IS2_SLAVEW    (0x1
> > << 6)
> > > +
> > > +#define
> > WL1273_IS2_TRI_AFTER_SENDING    (0x0 <<
> > 7)
> > > +#define
> > WL1273_IS2_TRI_ALWAYS_ACTIVE    (0x1 <<
> > 7)
> > > +
> > > +#define WL1273_IS2_SDOWS_RR    (0x0
> > << 8)
> > > +#define WL1273_IS2_SDOWS_RF    (0x1
> > << 8)
> > > +#define WL1273_IS2_SDOWS_FR    (0x2
> > << 8)
> > > +#define WL1273_IS2_SDOWS_FF    (0x3
> > << 8)
> > > +
> > > +#define WL1273_IS2_TRI_OPT    (0x0
> > << 10)
> > > +#define WL1273_IS2_TRI_ALWAYS    (0x1
> > << 10)
> > > +
> > > +#define WL1273_IS2_RATE_48K    (0x0
> > << 12)
> > > +#define WL1273_IS2_RATE_44_1K    (0x1
> > << 12)
> > > +#define WL1273_IS2_RATE_32K    (0x2
> > << 12)
> > > +#define WL1273_IS2_RATE_22_05K    (0x4
> > << 12)
> > > +#define WL1273_IS2_RATE_16K    (0x5
> > << 12)
> > > +#define WL1273_IS2_RATE_12K    (0x8
> > << 12)
> > > +#define WL1273_IS2_RATE_11_025    (0x9
> > << 12)
> > > +#define WL1273_IS2_RATE_8K    (0xa
> > << 12)
> > > +#define WL1273_IS2_RATE   
> >     (0xf << 12)
> > > +
> > > +#define WL1273_I2S_DEF_MODE   
> > (WL1273_IS2_WIDTH_32 | \
> > > +       
> >    
> >      WL1273_IS2_FORMAT_STD | \
> > > +       
> >    
> >      WL1273_IS2_MASTER | \
> > > +       
> >    
> >      WL1273_IS2_TRI_AFTER_SENDING |
> > \
> > > +       
> >    
> >      WL1273_IS2_SDOWS_RR | \
> > > +       
> >    
> >      WL1273_IS2_TRI_OPT | \
> > > +       
> >    
> >      WL1273_IS2_RATE_48K)
> > > +
> > > +/* Private IOCTL */
> > > +#define WL1273_CID_FM_BAND   
> > (V4L2_CID_PRIVATE_BASE + 2)
> > > +
> > > +#define SCHAR_MIN (-128)
> > > +#define SCHAR_MAX 127
> > > +
> > > +#define WL1273_FR_EVENT   
> >         (1 << 0)
> > > +#define WL1273_BL_EVENT   
> >         (1 << 1)
> > > +#define WL1273_RDS_EVENT   
> >     (1 << 2)
> > > +#define WL1273_BBLK_EVENT   
> >     (1 << 3)
> > > +#define WL1273_LSYNC_EVENT   
> >     (1 << 4)
> > > +#define WL1273_LEV_EVENT   
> >     (1 << 5)
> > > +#define WL1273_IFFR_EVENT   
> >     (1 << 6)
> > > +#define WL1273_PI_EVENT   
> >         (1 << 7)
> > > +#define WL1273_PD_EVENT   
> >         (1 << 8)
> > > +#define WL1273_STIC_EVENT   
> >     (1 << 9)
> > > +#define WL1273_MAL_EVENT   
> >     (1 << 10)
> > > +#define WL1273_POW_ENB_EVENT   
> >     (1 << 11)
> > > +#define WL1273_SCAN_OVER_EVENT   
> >     (1 << 12)
> > > +#define WL1273_ERROR_EVENT   
> >     (1 << 13)
> > > +
> > > +#define TUNER_MODE_STOP_SEARCH   
> >     0
> > > +#define TUNER_MODE_PRESET   
> >     1
> > > +#define TUNER_MODE_AUTO_SEEK   
> >     2
> > > +#define TUNER_MODE_AF   
> >         3
> > > +#define TUNER_MODE_AUTO_SEEK_PI   
> >     4
> > > +#define TUNER_MODE_AUTO_SEEK_BULK   
> > 5
> > > +
> > > +/* Allowed modes */
> > > +#define WL1273_RX_ALLOWED    0x01
> > > +#define WL1273_TX_ALLOWED    0x02
> > > +#define WL1273_RXTX_ALLOWED   
> > (WL1273_RX_ALLOWED | WL1273_TX_ALLOWED)
> > > +
> > > +struct band_info {
> > > +    u32 bottom_frequency;
> > > +    u32 top_frequency;
> > > +    u8 band;
> > > +};
> > > +
> > > +struct wl1273_fm_platform_data {
> > > +    int (*request_resources) (struct
> > i2c_client *client);
> > > +    void (*free_resources) (void);
> > > +    void (*enable) (void);
> > > +    void (*disable) (void);
> > > +
> > > +    u8 modes;
> > > +    unsigned int children;
> > > +};
> > > +
> > > +#define WL1273_FM_CORE_CELLS    2
> > > +
> > > +/* Allowed modes */
> > > +#define WL1273_RX_ALLOWED    0x01
> > > +#define WL1273_TX_ALLOWED    0x02
> > > +#define WL1273_RXTX_ALLOWED   
> > (WL1273_RX_ALLOWED | WL1273_TX_ALLOWED)
> > > +
> > > +struct wl1273_core {
> > > +    struct mfd_cell
> > cells[WL1273_FM_CORE_CELLS];
> > > +    struct i2c_client *i2c_dev;
> > > +
> > > +     u8 allowed_modes;
> > > +    unsigned int mode;
> > > +    unsigned int preemphasis;
> > > +    unsigned int audio_mode;
> > > +    unsigned int spacing;
> > > +    unsigned int tx_power;
> > > +    unsigned int rx_frequency;
> > > +    unsigned int tx_frequency;
> > > +    unsigned int band;
> > > +    unsigned int i2s_mode;
> > > +    unsigned int channel_number;
> > > +    unsigned int number_of_bands;
> > > +    unsigned int volume;
> > > +
> > > +    const struct band_info *bands;
> > > +
> > > +    /* RDS */
> > > +    bool rds_on;
> > > +    struct delayed_work work;
> > > +
> > > +    wait_queue_head_t read_queue;
> > > +    struct mutex lock; /* for
> > serializing fm radio operations */
> > > +    struct completion busy;
> > > +
> > > +    unsigned char *buffer;
> > > +    unsigned int buf_size;
> > > +    unsigned int rd_index;
> > > +    unsigned int wr_index;
> > > +
> > > +    /* Selected interrupts */
> > > +    u16 irq_flags;
> > > +    u16 irq_received;
> > > +};
> > > +
> > > +int wl1273_fm_write_cmd(struct wl1273_core
> *core, u8
> > cmd, u16 param);
> > > +int wl1273_fm_write_data(struct wl1273_core
> *core, u8
> > *data, u16 len);
> > > +int wl1273_fm_read_reg(struct wl1273_core *core,
> u8
> > reg, u16 *value);
> > > +
> > > +int wl1273_fm_set_audio(struct wl1273_core
> *core,
> > unsigned int mode);
> > > +int wl1273_fm_set_volume(struct wl1273_core
> *core,
> > unsigned int volume);
> > > +
> > > +#endif    /* ifndef RADIO_WL1273_H */
> > 
> > --
> > 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
> > 
> 
> 
> 


--
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


[Index of Archives]     [Linux Input]     [Video for Linux]     [Gstreamer Embedded]     [Mplayer Users]     [Linux USB Devel]     [Linux Audio Users]     [Linux Kernel]     [Linux SCSI]     [Yosemite Backpacking]
  Powered by Linux