summaryrefslogtreecommitdiff
path: root/sys/dev/usb
diff options
context:
space:
mode:
Diffstat (limited to 'sys/dev/usb')
-rw-r--r--sys/dev/usb/udcf.c130
1 files changed, 65 insertions, 65 deletions
diff --git a/sys/dev/usb/udcf.c b/sys/dev/usb/udcf.c
index 15f5a9de1d3..99248fccd31 100644
--- a/sys/dev/usb/udcf.c
+++ b/sys/dev/usb/udcf.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: udcf.c,v 1.4 2006/04/21 16:13:14 mbalmer Exp $ */
+/* $OpenBSD: udcf.c,v 1.5 2006/04/22 22:12:31 mbalmer Exp $ */
/*
* Copyright (c) 2006 Marc Balmer <mbalmer@openbsd.org>
@@ -29,6 +29,7 @@
#include <sys/time.h>
#include <sys/sensors.h>
+#include <dev/clock_subr.h>
#include <dev/usb/usb.h>
#include <dev/usb/usbdi.h>
#include <dev/usb/usbdi_util.h>
@@ -40,28 +41,9 @@ int udcfdebug = 0;
#else
#define DPRINTFN(n, x)
#endif
-#define DPRINTF(x) DPRINTFN(0, x)
-
-#define SECSPERMIN 60
-#define MINSPERHOUR 60
-#define HOURSPERDAY 24
-#define DAYSPERWEEK 7
-#define DAYSPERNYEAR 365
-#define DAYSPERLYEAR 366
-#define SECSPERHOUR (SECSPERMIN * MINSPERHOUR)
-#define SECSPERDAY ((long) SECSPERHOUR * HOURSPERDAY)
-#define MONSPERYEAR 12
-#define BASE_YEAR 2006
-#define JAN1_2006 1136073600L /* 2006/01/01 00:00:00 UTC */
-#define LEAPS_2006 486 /* num of leap years till 2006 */
-
-#define isleap(y) (((y) % 4) == 0 && (((y) % 100) != 0 || ((y) % 400) == 0))
-#define leaps(y) ((y) / 4 - (y) / 100 + (y) / 400 - LEAPS_2006)
-
-static const int mon_lengths[2][12] = {
- { 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 },
- { 31, 29, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 }
-};
+#define DPRINTF(x) DPRINTFN(0, x)
+
+#define SECSPERDAY ((long) 60 * 60 * 24)
#define UDCF_READ_REQ 0xc0
#define UDCF_READ_IDX 0x1f
@@ -84,9 +66,11 @@ struct udcf_softc {
struct timeout sc_db_to; /* debounce */
struct timeout sc_mg_to; /* minute-gap detect */
struct timeout sc_sl_to; /* signal-loss detect */
+ struct timeout sc_it_to; /* invalidate time */
struct usb_task sc_bv_task;
struct usb_task sc_mg_task;
struct usb_task sc_sl_task;
+ struct usb_task sc_it_task;
usb_device_request_t sc_req;
@@ -97,12 +81,13 @@ struct udcf_softc {
int sc_level;
time_t sc_last_mg;
- long sc_next; /* the time to become valid next */
+ struct timeval sc_last; /* when we last had a valid time */
+ time_t sc_next; /* the time to become valid next */
struct sensor sc_sensor;
};
-static int t1, t2, t3, t4, t5, t6, t7; /* timeouts in hz */
+static int t1, t2, t3, t4, t5, t6, t7, t8; /* timeouts in hz */
void udcf_intr(void *);
void udcf_probe(void *);
@@ -110,9 +95,11 @@ void udcf_probe(void *);
void udcf_bv_intr(void *);
void udcf_mg_intr(void *);
void udcf_sl_intr(void *);
+void udcf_it_intr(void *);
void udcf_bv_probe(void *);
void udcf_mg_probe(void *);
void udcf_sl_probe(void *);
+void udcf_it_probe(void *);
USB_DECLARE_DRIVER(udcf);
@@ -217,11 +204,13 @@ USB_ATTACH(udcf)
usb_init_task(&sc->sc_bv_task, udcf_bv_probe, sc);
usb_init_task(&sc->sc_mg_task, udcf_mg_probe, sc);
usb_init_task(&sc->sc_sl_task, udcf_sl_probe, sc);
+ usb_init_task(&sc->sc_it_task, udcf_it_probe, sc);
timeout_set(&sc->sc_to, udcf_intr, sc);
timeout_set(&sc->sc_bv_to, udcf_bv_intr, sc);
timeout_set(&sc->sc_mg_to, udcf_mg_intr, sc);
timeout_set(&sc->sc_sl_to, udcf_sl_intr, sc);
+ timeout_set(&sc->sc_it_to, udcf_it_intr, sc);
/* convert timevals to hz */
@@ -249,6 +238,9 @@ USB_ATTACH(udcf)
t.tv_sec = 8L;
t6 = tvtohz(&t);
+ t.tv_sec = SECSPERDAY;
+ t8 = tvtohz(&t);
+
/* Give the receiver some slack to stabilize */
timeout_add(&sc->sc_to, t3);
@@ -336,6 +328,18 @@ udcf_sl_intr(void *xsc)
}
/*
+ * invalidate time delta in the sensor if we did not receive time for
+ * more then MAXVALID seconds.
+ */
+
+void
+udcf_it_intr(void *xsc)
+{
+ struct udcf_softc *sc = xsc;
+ usb_add_task(sc->sc_udev, &sc->sc_it_task);
+}
+
+/*
* udcf_probe runs in a process context. If Bit 0 is set, the transmitter
* emits at full power. During the low-power emission we decode a zero bit.
*/
@@ -344,7 +348,6 @@ void
udcf_probe(void *xsc)
{
struct udcf_softc *sc = xsc;
- struct timeval now;
unsigned char data;
int actlen;
@@ -368,10 +371,15 @@ udcf_probe(void *xsc)
} else {
/* provide the time delta */
- microtime(&now);
- sc->sc_sensor.value = (now.tv_sec - sc->sc_next) * 1000 +
- now.tv_usec / 1000;
+ microtime(&sc->sc_last);
+ sc->sc_sensor.value = (sc->sc_last.tv_sec - sc->sc_next)
+ * 1000 + sc->sc_last.tv_usec / 1000;
sc->sc_sensor.status = SENSOR_S_OK;
+
+ /* retrigger the invalidation timer */
+
+ timeout_del(&sc->sc_it_to);
+ timeout_add(&sc->sc_it_to, t8);
}
sc->sc_tbits = 0LL;
sc->sc_mask = 1LL;
@@ -424,16 +432,13 @@ udcf_mg_probe(void *xsc)
{
struct udcf_softc *sc = xsc;
- int wday;
+ struct clock_ymdhms ymdhm;
int minute_bits, hour_bits, day_bits;
- int month_bits, year_bits;
+ int month_bits, year_bits, wday;
int p1, p2, p3;
int p1_bit, p2_bit, p3_bit;
int r_bit, a1_bit, a2_bit, z1_bit, z2_bit;
int s_bit, m_bit;
- int year, mon, mday, hour, min;
- int nyears, lyears, days, leap;
- int n;
u_int32_t parity = 0x6996;
@@ -445,7 +450,6 @@ udcf_mg_probe(void *xsc)
} else {
if (time_second - sc->sc_last_mg < 57) {
DPRINTF(("unexpected gap, resync\n"));
- sc->sc_sensor.status = SENSOR_S_UNKNOWN;
sc->sc_sync = 1;
timeout_del(&sc->sc_to);
timeout_add(&sc->sc_to, t5);
@@ -496,40 +500,22 @@ udcf_mg_probe(void *xsc)
/* Decode valid time */
- min = (minute_bits & 0x0f) +
- (minute_bits >> 4) * 10;
- hour = (hour_bits & 0x0f) +
- (hour_bits >> 4) * 10;
- mday = (day_bits & 0x0f) +
- (day_bits >> 4) * 10;
- mon = (month_bits & 0x0f) +
- (month_bits >> 4) * 10 - 1;
- year = 2000 + (year_bits & 0x0f) +
- (year_bits >> 4) * 10;
-
- /* convert to coordinated universal time */
- hour -= z1_bit ? 2 : 1;
- if (hour < 0)
- hour += 24;
+ ymdhm.dt_min = FROMBCD(minute_bits);
+ ymdhm.dt_hour = FROMBCD(hour_bits);
+ ymdhm.dt_day = FROMBCD(day_bits);
+ ymdhm.dt_mon = FROMBCD(month_bits);
+ ymdhm.dt_year = 2000 + FROMBCD(year_bits);
+ ymdhm.dt_sec = 0;
- if (year > BASE_YEAR)
- lyears = leaps(year - 1);
- else
- lyears = 0;
- nyears = year - BASE_YEAR - lyears;
- leap = isleap(year);
+ sc->sc_next = clock_ymdhms_to_secs(&ymdhm);
- days = nyears * DAYSPERNYEAR + lyears * DAYSPERLYEAR;
- for (n = 0; n < mon; n++)
- days += mon_lengths[leap][n];
- days += mday - 1;
+ /* convert to coordinated universal time */
- sc->sc_next = JAN1_2006 + days * SECSPERDAY +
- hour * SECSPERHOUR + min * SECSPERMIN;
+ sc->sc_next -= z1_bit ? 7200 : 3600;
DPRINTF(("\n%02d.%02d.%04d %02d:%02d:00 UTC",
- mday, mon + 1, year, hour,
- min));
+ ymdhm.dt_day, ymdhm.dt_mon + 1, ymdhm.dt_year,
+ ymdhm.dt_hour, ymdhm.dt_min));
DPRINTF((z1_bit ? ", dst" : ""));
DPRINTF((r_bit ? ", reserve antenna" : ""));
DPRINTF((a1_bit ? ", dst chg announced" : ""));
@@ -558,7 +544,6 @@ udcf_sl_probe(void *xsc)
return;
DPRINTF(("signal loss, resync\n"));
- sc->sc_sensor.status = SENSOR_S_UNKNOWN;
sc->sc_sync = 1;
timeout_del(&sc->sc_to);
timeout_add(&sc->sc_to, t5);
@@ -566,6 +551,21 @@ udcf_sl_probe(void *xsc)
timeout_add(&sc->sc_sl_to, t6);
}
+/* invalidate time delta */
+
+void
+udcf_it_probe(void *xsc)
+{
+ struct udcf_softc *sc = xsc;
+
+ if (sc->sc_dying)
+ return;
+
+ DPRINTF(("invalidating time delta\n"));
+ sc->sc_sensor.status = SENSOR_S_UNKNOWN;
+ sc->sc_sensor.value = 0LL;
+}
+
int
udcf_activate(device_ptr_t self, enum devact act)
{