Subject: Re: generic TODR changes for shark and ofppc
To: Garrett D'Amore <garrett_damore@tadpole.com>
From: Garrett D'Amore <garrett_damore@tadpole.com>
List: port-ofppc
Date: 09/12/2006 00:13:58
This is a multi-part message in MIME format.
--------------090900010404040503010107
Content-Type: text/plain; charset=ISO-8859-1
Content-Transfer-Encoding: 7bit
My apologies, the diff didn't have everything. I missed the most
important files, which was ofrtc, in a different directory. New diff
attached.
Garrett D'Amore wrote:
> Attached, find a diff that significantly changes the real-time-clock
> support for shark and ofppc.
>
> First, the old oftodclock.c hackery is gone, replaced by dev/ofw/ofrtc.
>
> The old ofrtc.c is changed, it no longer supports character entry
> points, but instead uses the new todr framework.
>
> One side effect of this is that ofppc should gain RTC clock support it
> lacked before.
>
> This has been tested and shown to work on shark.
>
>
> ------------------------------------------------------------------------
>
>
--------------090900010404040503010107
Content-Type: text/x-patch;
name="shark.diff"
Content-Transfer-Encoding: 7bit
Content-Disposition: inline;
filename="shark.diff"
Index: sys/arch/shark/conf/files.shark
===================================================================
RCS file: /cvsroot/src/sys/arch/shark/conf/files.shark,v
retrieving revision 1.10
diff -d -p -u -r1.10 files.shark
--- sys/arch/shark/conf/files.shark 11 Dec 2005 12:19:02 -0000 1.10
+++ sys/arch/shark/conf/files.shark 12 Sep 2006 07:13:05 -0000
@@ -43,7 +43,6 @@ include "dev/ofw/files.ofw"
include "arch/arm/ofw/files.ofw"
attach ofbus at root with ofbus_root
file arch/shark/ofw/ofw.c
-file arch/shark/ofw/oftodclock.c
# console initialization
file arch/shark/shark/consinit.c
Index: sys/arch/shark/include/types.h
===================================================================
RCS file: /cvsroot/src/sys/arch/shark/include/types.h,v
retrieving revision 1.3
diff -d -p -u -r1.3 types.h
--- sys/arch/shark/include/types.h 3 Sep 2006 13:51:23 -0000 1.3
+++ sys/arch/shark/include/types.h 12 Sep 2006 07:13:05 -0000
@@ -9,5 +9,6 @@
#define __ARM_FIQ_INDIRECT
#define __HAVE_DEVICE_REGISTER
+#define __HAVE_GENERIC_TODR
#endif
Index: sys/arch/shark/ofw/oftodclock.c
===================================================================
RCS file: sys/arch/shark/ofw/oftodclock.c
diff -N sys/arch/shark/ofw/oftodclock.c
--- sys/arch/shark/ofw/oftodclock.c 11 Dec 2005 12:19:05 -0000 1.5
+++ /dev/null 1 Jan 1970 00:00:00 -0000
@@ -1,290 +0,0 @@
-/* $NetBSD: oftodclock.c,v 1.5 2005/12/11 12:19:05 christos Exp $ */
-
-/*
- * Copyright 1997
- * Digital Equipment Corporation. All rights reserved.
- *
- * This software is furnished under license and may be used and
- * copied only in accordance with the following terms and conditions.
- * Subject to these conditions, you may download, copy, install,
- * use, modify and distribute this software in source and/or binary
- * form. No title or ownership is transferred hereby.
- *
- * 1) Any source code used, modified or distributed must reproduce
- * and retain this copyright notice and list of conditions as
- * they appear in the source file.
- *
- * 2) No right is granted to use any trade name, trademark, or logo of
- * Digital Equipment Corporation. Neither the "Digital Equipment
- * Corporation" name nor any trademark or logo of Digital Equipment
- * Corporation may be used to endorse or promote products derived
- * from this software without the prior written permission of
- * Digital Equipment Corporation.
- *
- * 3) This software is provided "AS-IS" and any express or implied
- * warranties, including but not limited to, any implied warranties
- * of merchantability, fitness for a particular purpose, or
- * non-infringement are disclaimed. In no event shall DIGITAL be
- * liable for any damages whatsoever, and in particular, DIGITAL
- * shall not be liable for special, indirect, consequential, or
- * incidental damages or damages for lost profits, loss of
- * revenue or loss of use, whether such damages arise in contract,
- * negligence, tort, under statute, in equity, at law or otherwise,
- * even if advised of the possibility of such damage.
- */
-
-/*
- * Real time clock (RTC) functions using OFW /rtc device:
- * inittodr()
- * resettodr()
- */
-
-#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: oftodclock.c,v 1.5 2005/12/11 12:19:05 christos Exp $");
-
-#include <sys/param.h>
-#include <sys/systm.h>
-#include <sys/time.h>
-#include <sys/kernel.h>
-
-#include <dev/ofw/openfirm.h>
-
-/* The OFW RTC interface, straight from wmb:
-
-selftest ( -- error? )
- Designed to be called from the "test" user interface command, e.g.
- test /rtc
- which displays a message if the test fails.
- Basically all this does is to check the battery.
-
-check-battery ( -- error? )
- Called from an open instance; returns non-zero if the battery
- is dead. Also displays a message to that effect, which it probably
- should not do.
-
-set-time ( s m h d m y -- )
-get-time ( -- s m h d m y )
- Called from an open instance.
- seconds: 0-59
- minutes: 0-59
- hours: 0-23
- day: 1-31
- month: 1-12
- year: e.g. 1997
-*/
-
-#define OFRTC_SEC 0
-#define OFRTC_MIN 1
-#define OFRTC_HR 2
-#define OFRTC_DOM 3
-#define OFRTC_MON 4
-#define OFRTC_YR 5
-
-static int month[12] = {31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31};
-
-#define SECMIN ((unsigned)60) /* seconds per minute */
-#define SECHOUR ((unsigned)(60*SECMIN)) /* seconds per hour */
-#define SECDAY ((unsigned)(24*SECHOUR)) /* seconds per day */
-#define SECYR ((unsigned)(365*SECDAY)) /* seconds per common year */
-
-static int yeartoday(int);
-static void setthetime(time_t, int);
-
-static int
-yeartoday(year)
- int year;
-{
-
- return ((year % 4) ? 365 : 366);
-}
-
-static int timeset = 0;
-
-static void
-setthetime(thetime, warning)
- time_t thetime;
- int warning;
-{
-
- timeset = 1;
- time.tv_sec = thetime;
- time.tv_usec = 0;
-
- if (warning)
- printf("WARNING: CHECK AND RESET THE DATE!\n");
-}
-
-static int ofrtc_phandle;
-static int ofrtc_ihandle = 0;
-static int ofrtcinited = 0;
-
-static void
-ofrtcinit(void)
-{
- char buf[256];
- int l;
- int chosen;
-
- if (ofrtcinited) return;
-
- if ((ofrtc_phandle = OF_finddevice("/rtc")) == -1)
- panic("OFW RTC: no package");
-
- if ((l = OF_getprop(ofrtc_phandle, "device_type", buf, sizeof buf - 1)) < 0)
- panic("OFW RTC: no device type");
-
- if ((l >= sizeof buf) || strcmp(buf, "rtc"))
- panic("OFW RTC: bad device type");
-
- if ((chosen = OF_finddevice("/chosen")) == -1 ||
- OF_getprop(chosen, "clock", &ofrtc_ihandle, sizeof(int)) < 0) {
- ofrtc_ihandle = 0;
- return;
- }
-
- ofrtc_ihandle = of_decode_int((unsigned char *)&ofrtc_ihandle);
-
- ofrtcinited = 1;
-}
-
-static int
-ofrtcstatus(void)
-{
- char status[256];
- int l;
-
- if ((ofrtc_ihandle == 0) || (l = OF_getprop(ofrtc_phandle, "status",
- status, sizeof status - 1)) < 0) {
- printf("OFW RTC: old firmware does not support RTC\n");
- return 0;
- }
-
- status[sizeof status - 1] = 0; /* always null terminate */
-
- if (strcmp(status, "okay")) { /* something is wrong */
- printf("RTC: %s\n", status);
- return 0;
- }
-
- return 1; /* all systems are go */
-}
-
-/*
- * Initialize the time of day register, based on the time base which is, e.g.
- * from a filesystem.
- */
-void
-inittodr(base)
- time_t base;
-{
- time_t n;
- int i, days = 0;
- int date[6];
- int yr;
-
- /*
- * We mostly ignore the suggested time and go for the RTC clock time
- * stored in the CMOS RAM. If the time can't be obtained from the
- * CMOS, or if the time obtained from the CMOS is 5 or more years
- * less than the suggested time, we used the suggested time. (In
- * the latter case, it's likely that the CMOS battery has died.)
- */
-
- if (base < 25*SECYR) { /* if before 1995, something's odd... */
- printf("WARNING: preposterous time in file system\n");
- /* read the system clock anyway */
- /* happy b-day sarina */
- base = 26*SECYR + 24*SECDAY + 18 * SECHOUR + 58 * SECMIN;
- }
-
- ofrtcinit();
-
- if (!ofrtcstatus()) {
- setthetime(base, 1);
- return;
- }
-
- if (OF_call_method("get-time", ofrtc_ihandle, 0, 6,
- date, date + 1, date + 2, date + 3, date + 4, date + 5)) {
- printf("OFW RTC: get-time failed\n");
- setthetime(base, 1);
- return;
- }
-
- n = date[OFRTC_SEC];
- n += date[OFRTC_MIN] * SECMIN;
- n += date[OFRTC_HR] * SECHOUR;
- n += (date[OFRTC_DOM] - 1) * SECDAY;
-
- yr = date[OFRTC_YR];
-
- if (yeartoday(yr) == 366)
- month[1] = 29;
- for (i = date[OFRTC_MON] - 2; i >= 0; i--)
- days += month[i];
- month[1] = 28;
- for (i = 1970; i < yr; i++)
- days += yeartoday(i);
- n += days * 3600 * 24;
-
- n += rtc_offset * 60;
-
- if (base < n - 5*SECYR)
- printf("WARNING: file system time much less than clock time\n");
-
- else if (base > n + 5*SECYR) {
- printf("WARNING: clock time much less than file system time\n");
- printf("WARNING: using file system time\n");
- setthetime(base, 1);
- return;
- }
-
- setthetime(n, 0);
-}
-
-/*
- * Reset the clock.
- */
-void
-resettodr(void)
-{
- time_t n;
- int diff, i, j;
- int sec, minute, hr, dom, mon, yr;
-
- /* old version of the firmware? */
- if (ofrtc_ihandle == 0) return;
-
- /*
- * We might have been called by boot() due to a crash early
- * on. Don't reset the clock chip in this case.
- */
- if (!timeset)
- return;
-
- diff = rtc_offset * 60;
- n = (time.tv_sec - diff) % (3600 * 24); /* hrs+mins+secs */
- sec = n % 60;
- n /= 60;
- minute = n % 60;
- hr = n / 60;
-
- n = (time.tv_sec - diff) / (3600 * 24); /* days */
-
- for (j = 1970, i = yeartoday(j); n >= i; j++, i = yeartoday(j))
- n -= i;
-
- yr = j;
-
- if (i == 366)
- month[1] = 29;
- for (i = 0; n >= month[i]; i++)
- n -= month[i];
- month[1] = 28;
- mon = ++i;
- dom = ++n;
-
- if (OF_call_method("set-time", ofrtc_ihandle, 6, 0,
- sec, minute, hr, dom, mon, yr))
- printf("OFW RTC: set-time failed\n");
-}
Index: sys/arch/ofppc/conf/majors.ofppc
===================================================================
RCS file: /cvsroot/src/sys/arch/ofppc/conf/majors.ofppc,v
retrieving revision 1.19
diff -d -p -u -r1.19 majors.ofppc
--- sys/arch/ofppc/conf/majors.ofppc 11 Dec 2005 12:18:29 -0000 1.19
+++ sys/arch/ofppc/conf/majors.ofppc 12 Sep 2006 07:13:05 -0000
@@ -12,7 +12,7 @@ device-major log char 5
device-major swap char 6 block 1 vmswap
device-major ofcons char 7 ofcons
device-major ofdisk char 8 block 0 ofdisk
-device-major ofrtc char 9 ofrtc
+#device-major ofrtc char 9 ofrtc
device-major bpf char 10 bpfilter
device-major rnd char 11 rnd
device-major raid char 12 block 2 raid
Index: sys/arch/ofppc/include/types.h
===================================================================
RCS file: /cvsroot/src/sys/arch/ofppc/include/types.h,v
retrieving revision 1.5
diff -d -p -u -r1.5 types.h
--- sys/arch/ofppc/include/types.h 22 Sep 2002 08:31:00 -0000 1.5
+++ sys/arch/ofppc/include/types.h 12 Sep 2006 07:13:05 -0000
@@ -3,3 +3,4 @@
#include <powerpc/types.h>
#define __HAVE_DEVICE_REGISTER
+#define __HAVE_GENERIC_TODR
Index: sys/arch/ofppc/ofppc/clock.c
===================================================================
RCS file: /cvsroot/src/sys/arch/ofppc/ofppc/clock.c,v
retrieving revision 1.12
diff -d -p -u -r1.12 clock.c
--- sys/arch/ofppc/ofppc/clock.c 24 Dec 2005 22:45:36 -0000 1.12
+++ sys/arch/ofppc/ofppc/clock.c 12 Sep 2006 07:13:05 -0000
@@ -50,28 +50,6 @@ static u_long ns_per_tick = 320;
long ticks_per_intr;
volatile u_long lasttb;
-/*
- * For now we let the machine run with boot time, not changing the clock
- * at inittodr at all.
- *
- * We might continue to do this due to setting up the real wall clock with
- * a user level utility in the future.
- */
-/* ARGSUSED */
-void
-inittodr(base)
- time_t base;
-{
-}
-
-/*
- * Similar to the above
- */
-void
-resettodr()
-{
-}
-
void
decr_intr(frame)
struct clockframe *frame;
Index: sys/dev/ofw/ofrtc.c
===================================================================
RCS file: /cvsroot/src/sys/dev/ofw/ofrtc.c,v
retrieving revision 1.19
diff -d -p -u -r1.19 ofrtc.c
--- sys/dev/ofw/ofrtc.c 29 Mar 2006 07:10:25 -0000 1.19
+++ sys/dev/ofw/ofrtc.c 12 Sep 2006 07:13:05 -0000
@@ -30,6 +30,38 @@
* OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
* ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
+/*
+ * Copyright 1997
+ * Digital Equipment Corporation. All rights reserved.
+ *
+ * This software is furnished under license and may be used and
+ * copied only in accordance with the following terms and conditions.
+ * Subject to these conditions, you may download, copy, install,
+ * use, modify and distribute this software in source and/or binary
+ * form. No title or ownership is transferred hereby.
+ *
+ * 1) Any source code used, modified or distributed must reproduce
+ * and retain this copyright notice and list of conditions as
+ * they appear in the source file.
+ *
+ * 2) No right is granted to use any trade name, trademark, or logo of
+ * Digital Equipment Corporation. Neither the "Digital Equipment
+ * Corporation" name nor any trademark or logo of Digital Equipment
+ * Corporation may be used to endorse or promote products derived
+ * from this software without the prior written permission of
+ * Digital Equipment Corporation.
+ *
+ * 3) This software is provided "AS-IS" and any express or implied
+ * warranties, including but not limited to, any implied warranties
+ * of merchantability, fitness for a particular purpose, or
+ * non-infringement are disclaimed. In no event shall DIGITAL be
+ * liable for any damages whatsoever, and in particular, DIGITAL
+ * shall not be liable for special, indirect, consequential, or
+ * incidental damages or damages for lost profits, loss of
+ * revenue or loss of use, whether such damages arise in contract,
+ * negligence, tort, under statute, in equity, at law or otherwise,
+ * even if advised of the possibility of such damage.
+ */
#include <sys/cdefs.h>
__KERNEL_RCSID(0, "$NetBSD: ofrtc.c,v 1.19 2006/03/29 07:10:25 thorpej Exp $");
@@ -40,31 +72,31 @@ __KERNEL_RCSID(0, "$NetBSD: ofrtc.c,v 1.
#include <sys/conf.h>
#include <sys/event.h>
+#include <dev/clock_subr.h>
#include <dev/ofw/openfirm.h>
+#define OFRTC_SEC 0
+#define OFRTC_MIN 1
+#define OFRTC_HR 2
+#define OFRTC_DOM 3
+#define OFRTC_MON 4
+#define OFRTC_YR 5
+
struct ofrtc_softc {
struct device sc_dev;
int sc_phandle;
int sc_ihandle;
+ struct todr_chip_handle sc_todr;
};
static int ofrtc_match(struct device *, struct cfdata *, void *);
static void ofrtc_attach(struct device *, struct device *, void *);
+static int ofrtc_gettod(todr_chip_handle_t, struct clock_ymdhms *);
+static int ofrtc_settod(todr_chip_handle_t, struct clock_ymdhms *);
CFATTACH_DECL(ofrtc, sizeof(struct ofrtc_softc),
ofrtc_match, ofrtc_attach, NULL, NULL);
-extern struct cfdriver ofrtc_cd;
-
-dev_type_open(ofrtc_open);
-dev_type_read(ofrtc_read);
-dev_type_write(ofrtc_write);
-
-const struct cdevsw ofrtc_cdevsw = {
- ofrtc_open, nullclose, ofrtc_read, ofrtc_write, noioctl,
- nostop, notty, nopoll, nommap, nokqfilter,
-};
-
static int
ofrtc_match(struct device *parent, struct cfdata *match, void *aux)
{
@@ -88,6 +120,7 @@ ofrtc_attach(struct device *parent, stru
struct ofrtc_softc *of = device_private(self);
struct ofbus_attach_args *oba = aux;
char name[32];
+ char path[256];
int l;
of->sc_phandle = oba->oba_phandle;
@@ -98,115 +131,74 @@ ofrtc_attach(struct device *parent, stru
if (l >= sizeof name)
l = sizeof name - 1;
name[l] = 0;
- printf(": %s\n", name);
-}
-int
-ofrtc_open(dev_t dev, int flags, int fmt, struct lwp *lwp)
-{
- struct ofrtc_softc *of;
- int unit = minor(dev);
- char path[256];
- int l;
-
- if (unit >= ofrtc_cd.cd_ndevs)
- return ENXIO;
- if (!(of = ofrtc_cd.cd_devs[unit]))
- return ENXIO;
if (!of->sc_ihandle) {
if ((l = OF_package_to_path(of->sc_phandle, path,
- sizeof path - 1)) < 0 ||
- l >= sizeof path)
- return ENXIO;
+ sizeof path - 1)) < 0 ||
+ l >= sizeof path) {
+ aprint_error(": cannot determine package path\n");
+ return;
+ }
path[l] = 0;
if (!(of->sc_ihandle = OF_open(path))) {
- if (of->sc_ihandle) {
- OF_close(of->sc_ihandle);
- of->sc_ihandle = 0;
- }
- return ENXIO;
- }
-
- }
-
- return 0;
-}
-
-static void
-twodigit(char *bp, int i)
-{
- *bp++ = i / 10 + '0';
- *bp = i % 10 + '0';
-}
-
-static int
-twodigits(char *bp)
-{
- int i;
+ aprint_error(": cannot open path\n");
+ return;
+ }
+ }
- i = *bp++ - '0';
- return i * 10 + *bp++ - '0';
+ of->sc_todr.todr_gettime_ymdhms = ofrtc_gettod;
+ of->sc_todr.todr_settime_ymdhms = ofrtc_settod;
+ of->sc_todr.cookie = of;
+ todr_attach(&of->sc_todr);
+ printf(": %s\n", name);
}
int
-ofrtc_read(dev_t dev, struct uio *uio, int flag)
+ofrtc_gettod(todr_chip_handle_t tch, struct clock_ymdhms *dt)
{
- struct ofrtc_softc *of = ofrtc_cd.cd_devs[minor(dev)];
+ struct ofrtc_softc *of = tch->cookie;
int date[6];
- char buf[14];
- int xlen;
- if (uio->uio_offset >= sizeof buf)
- return 0;
+ /*
+ * We mostly ignore the suggested time and go for the RTC clock time
+ * stored in the CMOS RAM. If the time can't be obtained from the
+ * CMOS, or if the time obtained from the CMOS is 5 or more years
+ * less than the suggested time, we used the suggested time. (In
+ * the latter case, it's likely that the CMOS battery has died.)
+ */
if (OF_call_method("get-time", of->sc_ihandle, 0, 6,
- date, date + 1, date + 2,
- date + 3, date + 4, date + 5))
+ date, date + 1, date + 2, date + 3, date + 4, date + 5)) {
return EIO;
+ }
- twodigit(buf, date[5] % 100);
- twodigit(buf + 2, date[4]);
- twodigit(buf + 4, date[3]);
- twodigit(buf + 6, date[2]);
- twodigit(buf + 8, date[1]);
- buf[10] = '.';
- twodigit(buf + 11, date[0]);
- buf[13] = '\n';
-
- xlen = sizeof(buf) - uio->uio_offset;
- if (xlen > uio->uio_resid)
- xlen = uio->uio_resid;
+ dt->dt_sec = date[OFRTC_SEC];
+ dt->dt_min = date[OFRTC_MIN];
+ dt->dt_hour = date[OFRTC_HR];
+ dt->dt_day = date[OFRTC_DOM];
+ dt->dt_mon = date[OFRTC_MON];
+ dt->dt_year = date[OFRTC_YR];
- return uiomove((caddr_t)buf, xlen, uio);
+ return 0;
}
int
-ofrtc_write(dev_t dev, struct uio *uio, int flag)
+ofrtc_settod(todr_chip_handle_t tch, struct clock_ymdhms *dt)
{
- struct ofrtc_softc *of = ofrtc_cd.cd_devs[minor(dev)];
- char buf[14];
- int cnt, year, error;
-
- /*
- * We require atomic updates!
- */
- cnt = uio->uio_resid;
- if (uio->uio_offset || (cnt != sizeof buf && cnt != sizeof buf - 1))
- return EINVAL;
-
- if ((error = uiomove((caddr_t)buf, sizeof buf, uio)) != 0)
- return error;
+ struct ofrtc_softc *of = tch->cookie;
+ int sec, minute, hr, dom, mon, yr;
- if (cnt == sizeof buf && buf[sizeof buf - 1] != '\n')
- return EINVAL;
+ sec = dt->dt_sec;
+ minute = dt->dt_min;
+ hr = dt->dt_hour;
+ dom = dt->dt_day;
+ mon = dt->dt_mon;
+ yr = dt->dt_year;
- year = twodigits(buf) + 1900;
- if (year < 1970)
- year += 100;
if (OF_call_method("set-time", of->sc_ihandle, 6, 0,
- twodigits(buf + 11), twodigits(buf + 8), twodigits(buf + 6),
- twodigits(buf + 4), twodigits(buf + 2), year))
+ sec, minute, hr, dom, mon, yr))
return EIO;
+
return 0;
}
--------------090900010404040503010107--