Google Groups no longer supports new Usenet posts or subscriptions. Historical content remains viewable.
Dismiss

[PATCH][I2C] ST M41T00 I2C RTC chip driver

21 views
Skip to first unread message

Mark A. Greer

unread,
Jan 31, 2005, 1:09:48 PM1/31/05
to Greg KH, LM Sensors, lkml
This patch adds support for the ST M41T00 RTC chip.

You will likely notice that it implements a PPC-specific interface
(/dev/rtc->drivers/char/genrtc.h->include/asm-ppc/rtc.h->this file).
This was necessary to support a subset of ppc platforms that need to
hook up the rtc support at runtime. If I implemented /dev/rtc directly
or interfaced to genrtc.c directly, those platforms couldn't use this
driver. Eventually, I hope to work on more uniform rtc support across
all the processor architectures.

Also, on ppc at least, the hw clock can be set from a timer interrupt if
STA_UNSYNC is not set (e.g., ntpd is running). To handle this, a
tasklet is used to set the clock if in_interrupt() is true.

I'd appreciate an comments or to have it pushed into the kernel.org tree
if its acceptable.

Thanks,

Mark

Signed-off-by: Mark A. Greer <mgr...@mvista.com>
--

m41t00.patch

linux-os

unread,
Jan 31, 2005, 1:46:00 PM1/31/05
to Mark A. Greer, Greg KH, LM Sensors, lkml

On ix86 machines, it is appropriate to read the RTC clock
several times in a row until nothing changes. This protects
against getting bad readings when some values wrap (like
seconds). You can't stop the clock when you read it
or you will lose time. I don't see anything like
this in your code. Also, when setting the clock it
is necessary to stop the clock so its settings don't
change while you are writing a new time. I also don't
see anything like this in your code either.

Also ix86 machines have a spin-lock, rtc_lock, so that
other procedures, even interrupts can access its registers.
I see you using a semaphore that can't be used in interrupt
context.

Notice: the RTC clock is used for high-precision timing
via interrupt in some ix86 drivers. If you modify the
RTC code on all platforms as you propose, you cannot
"keep" the RTC all to your self. Its interrupt must
be available to drivers and access to the hardware needs
to be locked with the existing spin-lock.

I note that on 2.6.9 on somebody broke it so that, the interrupt is
used for something only ONCE during startup and ONCE during shut-down.

CPU0
0: 6225448 IO-APIC-edge timer
1: 14002 IO-APIC-edge i8042
8: 1 IO-APIC-edge rtc <--- bingo
9: 0 IO-APIC-level acpi
[snipped...]

I don't have a clue why anybody would grab that interrupt. Fortunately,
the interrupt-grabbing code can be loaded as a module and then
unloaded so that other drivers can use that device and its interrupt.
Any changes to the RTC code need to consider these things.

Cheers,
Dick Johnson
Penguin : Linux version 2.6.9 on an i686 machine (5537.79 BogoMips).
Notice : All mail here is now cached for review by Dictator Bush.
98.36% of all statistics are fiction.
-
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majo...@vger.kernel.org
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/

Jean Delvare

unread,
Jan 31, 2005, 3:08:32 PM1/31/05
to Mark A. Greer, Greg KH, LM Sensors, lkml
Hi Mark,

> This patch adds support for the ST M41T00 RTC chip.

As for your other driver, I lack the device-specific knowledge to
comment on the functionality, but still have some technical comments on
the code itself:

> + This driver can also be built as a module. If so, the module
> + will be called i2c-m41t00.

It'll actually be called m41t00, according to the Makefile.

> +struct m41t00_data {
> + struct i2c_client client;
> +};

You don't have to do that. Including the i2c_client stucture in the data
structure is a trick which let us get both allocated with a single
kmalloc (and freed with a single kfree) while still respecting the
arch-dependent alignment requirements. If you have no private data to
carry around, you can do the kmalloc on the i2c_client structure
directly, and have client->data point to NULL (which it actually already
does thanks to memset). This will save some code in both the detection
and the detach functions.

However, if you know that, in a future update of this driver, you *will*
have to store client-private data, then I guess you can keep it this way.

> + i2c_detach_client(client);

This one supposedly can fail.

> + .name = "M41T00",

No caps in name please (will be used in sysfs).

> +static int __devinit
> +m41t00_init(void)
> +{
> + return i2c_add_driver(&m41t00_driver);
> +}
>
> +static void __devexit
> +m41t00_exit(void)
> +{
> + i2c_del_driver(&m41t00_driver);
> + return;
> +}

Should be __init and __exit, respectively, unless I am mistaken. And the
last return is usless.

I'm also suspicious about the other __devexit and __devinit you used. No
other i2c chip drivers has them.

Thanks,
--
Jean Delvare

Mark A. Greer

unread,
Jan 31, 2005, 3:56:21 PM1/31/05
to linu...@analogic.com, Greg KH, LM Sensors, lkml
linux-os wrote:

>
> On ix86 machines, it is appropriate to read the RTC clock
> several times in a row until nothing changes. This protects
> against getting bad readings when some values wrap (like
> seconds). You can't stop the clock when you read it
> or you will lose time. I don't see anything like this in your code.
> Also, when setting the clock it
> is necessary to stop the clock so its settings don't
> change while you are writing a new time. I also don't
> see anything like this in your code either.


This particular RTC chip provide no mechanism to manually stop the clock
for reading or writing. However, when you begin reading the clock
registers, it delays updating the externally visible register values for
250ms but contiues to keep time internally so time isn't lost. I can't
find anything in the manual WRT updating the clock.

There is a problem that if all the register reads don't happen within
the 250ms window, it could return a bad value if a register wraps.
Unless someone else points out why this is a bad idea, I'll add a loop
to ensure the same values were read twice in a row.

>
>
> Also ix86 machines have a spin-lock, rtc_lock, so that
> other procedures, even interrupts can access its registers.
> I see you using a semaphore that can't be used in interrupt
> context.


This is an I2C based RTC chip and the I2C subsystem used to access it
assumes (AFAICT) that its not called from an interrupt handler (e.g.,
drivers/i2c/i2c-core.c:i2c_transfer calls() calls down()/up()). So I
need to handle an interrupt handler calling this driver which then calls
into code that assumes its not in an interrupt handler. That's why the
'set' routine schedules a tasklet if its in an interrupt handler. In
ppc, at least, the 'get' routine isn't called from an interrupt so its
not an issue.

>
>
> Notice: the RTC clock is used for high-precision timing
> via interrupt in some ix86 drivers. If you modify the
> RTC code on all platforms as you propose, you cannot
> "keep" the RTC all to your self. Its interrupt must
> be available to drivers and access to the hardware needs
> to be locked with the existing spin-lock.


Hmm, interesting. Note that this RTC doesn't generate an
interrupt/initiate an i2c transaction.

>
>
> I note that on 2.6.9 on somebody broke it so that, the interrupt is
> used for something only ONCE during startup and ONCE during shut-down.
>
> CPU0
> 0: 6225448 IO-APIC-edge timer
> 1: 14002 IO-APIC-edge i8042
> 8: 1 IO-APIC-edge rtc <--- bingo
> 9: 0 IO-APIC-level acpi
> [snipped...]
>
> I don't have a clue why anybody would grab that interrupt. Fortunately,
> the interrupt-grabbing code can be loaded as a module and then
> unloaded so that other drivers can use that device and its interrupt.
> Any changes to the RTC code need to consider these things.


Okay, thanks for the insight.

Mark

Mark A. Greer

unread,
Feb 1, 2005, 3:46:40 PM2/1/05
to LM Sensors, LKML, Greg KH
Thank you for your comments, Jean.

Jean Delvare wrote:

>Hi Mark,


>
>
>
>>+ This driver can also be built as a module. If so, the module
>>+ will be called i2c-m41t00.
>>
>>
>
>It'll actually be called m41t00, according to the Makefile.
>

Good catch.

>
>
>>+struct m41t00_data {
>>+ struct i2c_client client;
>>+};
>>
>>
>
>You don't have to do that. Including the i2c_client stucture in the data
>structure is a trick which let us get both allocated with a single
>kmalloc (and freed with a single kfree) while still respecting the
>arch-dependent alignment requirements. If you have no private data to
>carry around, you can do the kmalloc on the i2c_client structure
>directly, and have client->data point to NULL (which it actually already
>does thanks to memset). This will save some code in both the detection
>and the detach functions.
>
>However, if you know that, in a future update of this driver, you *will*
>have to store client-private data, then I guess you can keep it this way.
>

Its gone.

>>+ i2c_detach_client(client);
>>
>>
>
>This one supposedly can fail.
>

Okay, I'll check.

>
>
>>+ .name = "M41T00",
>>
>>
>
>No caps in name please (will be used in sysfs).
>

Done.

>
>
>
>>+static int __devinit
>>+m41t00_init(void)
>>+{
>>+ return i2c_add_driver(&m41t00_driver);
>>+}
>>
>>+static void __devexit
>>+m41t00_exit(void)
>>+{
>>+ i2c_del_driver(&m41t00_driver);
>>+ return;
>>+}
>>
>>
>
>Should be __init and __exit, respectively, unless I am mistaken. And the
>last return is usless.
>

I thought the __devxxx ones were the best ones to use but I don't know
for sure. I'll change them to __init/__exit.

>
>I'm also suspicious about the other __devexit and __devinit you used. No
>other i2c chip drivers has them.
>
>

Done.

Here is a replacement patch that should address Jean Delvare and Dick
Johnson's issues. Please let me know if there are any more issues.

m41t00_2.patch

Mark A. Greer

unread,
Feb 4, 2005, 6:24:50 PM2/4/05
to LM Sensors, LKML, Greg KH
From http://archives.andrew.net.au/lm-sensors/msg29319.html:

Mark A. Greer wrote:

> <snip>


>
> Here is a replacement patch that should address Jean Delvare and Dick
> Johnson's issues. Please let me know if there are any more issues.
>
> Thanks,
>
> Mark
>
> Signed-off-by: Mark A. Greer <mgr...@mvista.com>
> --


I haven't heard of any more problems so can I get this patch applied?

Thanks,

Greg KH

unread,
Feb 4, 2005, 6:29:59 PM2/4/05
to Mark A. Greer, LM Sensors, LKML
On Fri, Feb 04, 2005 at 04:14:51PM -0700, Mark A. Greer wrote:
> From http://archives.andrew.net.au/lm-sensors/msg29319.html:
>
> Mark A. Greer wrote:
>
> ><snip>
> >
> >Here is a replacement patch that should address Jean Delvare and Dick
> >Johnson's issues. Please let me know if there are any more issues.
> >
> >Thanks,
> >
> >Mark
> >
> >Signed-off-by: Mark A. Greer <mgr...@mvista.com>
> >--
>
>
> I haven't heard of any more problems so can I get this patch applied?

Can you resend it with a proper Changelog description in the top of the
email and the signed-off-by line? thanks,

greg k-h

Mark A. Greer

unread,
Feb 4, 2005, 7:22:07 PM2/4/05
to Greg KH, LM Sensors, LKML
Greg KH wrote:

>Can you resend it with a proper Changelog description in the top of the
>email and the signed-off-by line? thanks,
>
>greg k-h
>
>
>

Certainly.
--

This patch adds support for the ST M41T00 I2C RTC chip.

This rtc chip has no mechanism to freeze it's registers while being
read; however, it will delay updating the external values of the
registers for 250ms after a register is read. To ensure that a sane
time value is read, the driver verifies that the same registers values
were read twice before returning.

Also, when setting the rtc from an interrupt handler, a tasklet is used
to provide the context required by the i2c core code.

Please apply.

m41t00_3.patch

Greg KH

unread,
Feb 17, 2005, 5:43:43 PM2/17/05
to Mark A. Greer, LM Sensors, LKML

Applied, thanks.

0 new messages