Commit 2fb07a10 authored by Felipe Balbi's avatar Felipe Balbi Committed by Alexandre Belloni

rtc: ds1307: Convert to threaded IRQ

The driver currently emulates the concept of threaded IRQ using a
workqueue, which it really does not need to. Instead, switch over to
threaded_irq handlers which is meant precisely for the same purpose.
Signed-off-by: default avatarFelipe Balbi <balbi@ti.com>
Signed-off-by: default avatarNishanth Menon <nm@ti.com>
Reviewed-by: default avatarGrygorii Strashko <grygorii.strashko@ti.com>
Signed-off-by: default avatarAlexandre Belloni <alexandre.belloni@free-electrons.com>
parent f377ea88
...@@ -114,7 +114,6 @@ struct ds1307 { ...@@ -114,7 +114,6 @@ struct ds1307 {
#define HAS_ALARM 1 /* bit 1 == irq claimed */ #define HAS_ALARM 1 /* bit 1 == irq claimed */
struct i2c_client *client; struct i2c_client *client;
struct rtc_device *rtc; struct rtc_device *rtc;
struct work_struct work;
s32 (*read_block_data)(const struct i2c_client *client, u8 command, s32 (*read_block_data)(const struct i2c_client *client, u8 command,
u8 length, u8 *values); u8 length, u8 *values);
s32 (*write_block_data)(const struct i2c_client *client, u8 command, s32 (*write_block_data)(const struct i2c_client *client, u8 command,
...@@ -311,27 +310,17 @@ static s32 ds1307_native_smbus_read_block_data(const struct i2c_client *client, ...@@ -311,27 +310,17 @@ static s32 ds1307_native_smbus_read_block_data(const struct i2c_client *client,
/*----------------------------------------------------------------------*/ /*----------------------------------------------------------------------*/
/* /*
* The IRQ logic includes a "real" handler running in IRQ context just
* long enough to schedule this workqueue entry. We need a task context
* to talk to the RTC, since I2C I/O calls require that; and disable the
* IRQ until we clear its status on the chip, so that this handler can
* work with any type of triggering (not just falling edge).
*
* The ds1337 and ds1339 both have two alarms, but we only use the first * The ds1337 and ds1339 both have two alarms, but we only use the first
* one (with a "seconds" field). For ds1337 we expect nINTA is our alarm * one (with a "seconds" field). For ds1337 we expect nINTA is our alarm
* signal; ds1339 chips have only one alarm signal. * signal; ds1339 chips have only one alarm signal.
*/ */
static void ds1307_work(struct work_struct *work) static irqreturn_t ds1307_irq(int irq, void *dev_id)
{ {
struct ds1307 *ds1307; struct i2c_client *client = dev_id;
struct i2c_client *client; struct ds1307 *ds1307 = i2c_get_clientdata(client);
struct mutex *lock; struct mutex *lock = &ds1307->rtc->ops_lock;
int stat, control; int stat, control;
ds1307 = container_of(work, struct ds1307, work);
client = ds1307->client;
lock = &ds1307->rtc->ops_lock;
mutex_lock(lock); mutex_lock(lock);
stat = i2c_smbus_read_byte_data(client, DS1337_REG_STATUS); stat = i2c_smbus_read_byte_data(client, DS1337_REG_STATUS);
if (stat < 0) if (stat < 0)
...@@ -352,18 +341,8 @@ static void ds1307_work(struct work_struct *work) ...@@ -352,18 +341,8 @@ static void ds1307_work(struct work_struct *work)
} }
out: out:
if (test_bit(HAS_ALARM, &ds1307->flags))
enable_irq(client->irq);
mutex_unlock(lock); mutex_unlock(lock);
}
static irqreturn_t ds1307_irq(int irq, void *dev_id)
{
struct i2c_client *client = dev_id;
struct ds1307 *ds1307 = i2c_get_clientdata(client);
disable_irq_nosync(irq);
schedule_work(&ds1307->work);
return IRQ_HANDLED; return IRQ_HANDLED;
} }
...@@ -634,13 +613,14 @@ static const struct rtc_class_ops ds13xx_rtc_ops = { ...@@ -634,13 +613,14 @@ static const struct rtc_class_ops ds13xx_rtc_ops = {
MCP794XX_BIT_ALMX_C1 | \ MCP794XX_BIT_ALMX_C1 | \
MCP794XX_BIT_ALMX_C2) MCP794XX_BIT_ALMX_C2)
static void mcp794xx_work(struct work_struct *work) static irqreturn_t mcp794xx_irq(int irq, void *dev_id)
{ {
struct ds1307 *ds1307 = container_of(work, struct ds1307, work); struct i2c_client *client = dev_id;
struct i2c_client *client = ds1307->client; struct ds1307 *ds1307 = i2c_get_clientdata(client);
struct mutex *lock = &ds1307->rtc->ops_lock;
int reg, ret; int reg, ret;
mutex_lock(&ds1307->rtc->ops_lock); mutex_lock(lock);
/* Check and clear alarm 0 interrupt flag. */ /* Check and clear alarm 0 interrupt flag. */
reg = i2c_smbus_read_byte_data(client, MCP794XX_REG_ALARM0_CTRL); reg = i2c_smbus_read_byte_data(client, MCP794XX_REG_ALARM0_CTRL);
...@@ -665,9 +645,9 @@ static void mcp794xx_work(struct work_struct *work) ...@@ -665,9 +645,9 @@ static void mcp794xx_work(struct work_struct *work)
rtc_update_irq(ds1307->rtc, 1, RTC_AF | RTC_IRQF); rtc_update_irq(ds1307->rtc, 1, RTC_AF | RTC_IRQF);
out: out:
if (test_bit(HAS_ALARM, &ds1307->flags)) mutex_unlock(lock);
enable_irq(client->irq);
mutex_unlock(&ds1307->rtc->ops_lock); return IRQ_HANDLED;
} }
static int mcp794xx_read_alarm(struct device *dev, struct rtc_wkalrm *t) static int mcp794xx_read_alarm(struct device *dev, struct rtc_wkalrm *t)
...@@ -896,6 +876,8 @@ static int ds1307_probe(struct i2c_client *client, ...@@ -896,6 +876,8 @@ static int ds1307_probe(struct i2c_client *client,
bool want_irq = false; bool want_irq = false;
unsigned char *buf; unsigned char *buf;
struct ds1307_platform_data *pdata = dev_get_platdata(&client->dev); struct ds1307_platform_data *pdata = dev_get_platdata(&client->dev);
irq_handler_t irq_handler = ds1307_irq;
static const int bbsqi_bitpos[] = { static const int bbsqi_bitpos[] = {
[ds_1337] = 0, [ds_1337] = 0,
[ds_1339] = DS1339_BIT_BBSQI, [ds_1339] = DS1339_BIT_BBSQI,
...@@ -962,8 +944,6 @@ static int ds1307_probe(struct i2c_client *client, ...@@ -962,8 +944,6 @@ static int ds1307_probe(struct i2c_client *client,
* running on Vbackup (BBSQI/BBSQW) * running on Vbackup (BBSQI/BBSQW)
*/ */
if (ds1307->client->irq > 0 && chip->alarm) { if (ds1307->client->irq > 0 && chip->alarm) {
INIT_WORK(&ds1307->work, ds1307_work);
ds1307->regs[0] |= DS1337_BIT_INTCN ds1307->regs[0] |= DS1337_BIT_INTCN
| bbsqi_bitpos[ds1307->type]; | bbsqi_bitpos[ds1307->type];
ds1307->regs[0] &= ~(DS1337_BIT_A2IE | DS1337_BIT_A1IE); ds1307->regs[0] &= ~(DS1337_BIT_A2IE | DS1337_BIT_A1IE);
...@@ -1053,7 +1033,7 @@ static int ds1307_probe(struct i2c_client *client, ...@@ -1053,7 +1033,7 @@ static int ds1307_probe(struct i2c_client *client,
case mcp794xx: case mcp794xx:
rtc_ops = &mcp794xx_rtc_ops; rtc_ops = &mcp794xx_rtc_ops;
if (ds1307->client->irq > 0 && chip->alarm) { if (ds1307->client->irq > 0 && chip->alarm) {
INIT_WORK(&ds1307->work, mcp794xx_work); irq_handler = mcp794xx_irq;
want_irq = true; want_irq = true;
} }
break; break;
...@@ -1176,8 +1156,9 @@ static int ds1307_probe(struct i2c_client *client, ...@@ -1176,8 +1156,9 @@ static int ds1307_probe(struct i2c_client *client,
} }
if (want_irq) { if (want_irq) {
err = request_irq(client->irq, ds1307_irq, IRQF_SHARED, err = request_threaded_irq(client->irq, NULL, irq_handler,
ds1307->rtc->name, client); IRQF_SHARED | IRQF_ONESHOT,
ds1307->rtc->name, client);
if (err) { if (err) {
client->irq = 0; client->irq = 0;
dev_err(&client->dev, "unable to request IRQ!\n"); dev_err(&client->dev, "unable to request IRQ!\n");
...@@ -1231,10 +1212,8 @@ static int ds1307_remove(struct i2c_client *client) ...@@ -1231,10 +1212,8 @@ static int ds1307_remove(struct i2c_client *client)
{ {
struct ds1307 *ds1307 = i2c_get_clientdata(client); struct ds1307 *ds1307 = i2c_get_clientdata(client);
if (test_and_clear_bit(HAS_ALARM, &ds1307->flags)) { if (test_and_clear_bit(HAS_ALARM, &ds1307->flags))
free_irq(client->irq, client); free_irq(client->irq, client);
cancel_work_sync(&ds1307->work);
}
if (test_and_clear_bit(HAS_NVRAM, &ds1307->flags)) if (test_and_clear_bit(HAS_NVRAM, &ds1307->flags))
sysfs_remove_bin_file(&client->dev.kobj, ds1307->nvram); sysfs_remove_bin_file(&client->dev.kobj, ds1307->nvram);
......
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