summaryrefslogtreecommitdiff
path: root/sys/kern/tty_nmea.c
diff options
context:
space:
mode:
authorMarc Balmer <mbalmer@cvs.openbsd.org>2006-06-13 07:02:00 +0000
committerMarc Balmer <mbalmer@cvs.openbsd.org>2006-06-13 07:02:00 +0000
commit2b75dad143f49702efaf7b7b9de4b2668188654a (patch)
tree24d25381ed8e43631f6fd002403e6b0e6b3e89bb /sys/kern/tty_nmea.c
parent788909a5b70aea7867b5347fd4ae7a7401833f31 (diff)
Next round of clean-up: Remove the ugly state machine and replace it with
a simpler function that splits the nmea sentence into fields, verifies the checksum and eventually decodes the data (if it's a GPRMC sentence). Ignore Loran-C for the moment, no one has such a receiver. Loran-C can easily be added later if we have a receiver to test with.
Diffstat (limited to 'sys/kern/tty_nmea.c')
-rw-r--r--sys/kern/tty_nmea.c290
1 files changed, 106 insertions, 184 deletions
diff --git a/sys/kern/tty_nmea.c b/sys/kern/tty_nmea.c
index 87685d82641..d0c10166e6c 100644
--- a/sys/kern/tty_nmea.c
+++ b/sys/kern/tty_nmea.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: tty_nmea.c,v 1.6 2006/06/10 23:15:26 mbalmer Exp $ */
+/* $OpenBSD: tty_nmea.c,v 1.7 2006/06/13 07:01:59 mbalmer Exp $ */
/*
* Copyright (c) 2006 Marc Balmer <mbalmer@openbsd.org>
@@ -16,7 +16,7 @@
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
*/
-/* line discipline to decode NMEA0183 data */
+/* line discipline to decode NMEA 0183 data */
#include <sys/param.h>
#include <sys/systm.h>
@@ -53,45 +53,22 @@ int nmeainput(int, struct tty *);
void nmeaattach(int);
#define NMEAMAX 82
-#define MAXFLDS 12
-
-/* NMEA talker identifiers */
-#define TI_UNK 0
-#define TI_GPS 1
-#define TI_LORC 2
-
-/* NMEA message types */
-#define MSG_RMC 5393731 /* recommended minimum sentence C */
-#define MSG_GGA 4671297
-#define MSG_GSA 4674369 /* satellites active */
-#define MSG_GSV 4674390 /* satellites in view */
-#define MSG_VTG 5657671 /* velocity, direction, speed */
+#define MAXFLDS 16
int nmea_count = 0;
struct nmea {
char cbuf[NMEAMAX];
struct sensor time;
- struct timeval tv; /* soft timestamp */
struct timespec ts;
int64_t last; /* last time rcvd */
- int state; /* state we're in */
+ int sync;
int pos;
- int fpos[MAXFLDS];
- int flds; /* expect nr of fields */
- int fldcnt; /* actual count of fields */
- int cksum; /* calculated checksum */
- int msgcksum; /* received cksum */
- int ti; /* talker identifier */
- int msg; /* NMEA msg type */
};
-/* NMEA protocol state machine */
-void nmea_hdlr(struct nmea *, int c);
-
/* NMEA decoding */
-void nmea_decode(struct nmea *);
-void nmea_rmc(struct nmea *);
+void nmea_scan(struct nmea *);
+void nmea_gprmc(struct nmea *, char *fld[], int fldcnt);
/* date and time conversion */
int nmea_date_to_nano(char *s, int64_t *nano);
@@ -99,18 +76,6 @@ int nmea_time_to_nano(char *s, int64_t *nano);
static inline int leapyear(int year);
int nmea_ymd_to_secs(int year, int month, int day, time_t *secs);
-/* helper functions */
-void nmea_bufadd(struct nmea *, int);
-
-enum states {
- S_SYNC = 0,
- S_TI_1,
- S_TI_2,
- S_MSG,
- S_DATA,
- S_CKSUM
-};
-
void
nmeaattach(int dummy)
{
@@ -133,7 +98,7 @@ nmeaopen(dev_t dev, struct tty *tp)
nmea_count++);
np->time.status = SENSOR_S_UNKNOWN;
np->time.type = SENSOR_TIMEDELTA;
- np->state = S_SYNC;
+ np->sync = 1;
np->time.flags = SENSOR_FINVALID;
sensor_add(&np->time);
tp->t_sc = (caddr_t)np;
@@ -159,185 +124,142 @@ nmeaclose(struct tty *tp, int flags)
return linesw[TTYDISC].l_close(tp, flags);
}
-/* scan input from tty for NMEA sentences */
+/* collect NMEA sentence from tty */
int
nmeainput(int c, struct tty *tp)
{
struct nmea *np = (struct nmea *)tp->t_sc;
- nmea_hdlr(np, c);
+ switch (c) {
+ case '$':
+ /* timestamp and delta refs now */
+ nanotime(&np->ts);
+ np->pos = 0;
+ np->sync = 0;
+ break;
+ case '\r':
+ case '\n':
+ if (!np->sync) {
+ np->cbuf[np->pos] = '\0';
+ nmea_scan(np);
+ np->sync = 1;
+ }
+ break;
+ default:
+ if (!np->sync && np->pos < (NMEAMAX - 1))
+ np->cbuf[np->pos++] = c;
+ break;
+ }
/* pass data to termios */
return linesw[TTYDISC].l_rint(c, tp);
}
-/* NMEA state machine */
+/* Scan the NMEA sentence just received */
void
-nmea_hdlr(struct nmea *np, int c)
+nmea_scan(struct nmea *np)
{
- switch (np->state) {
- case S_SYNC:
- switch (c) {
- case '$':
- /* timestamp and delta refs now */
- microtime(&np->tv);
- nanotime(&np->ts);
- np->pos = 0;
- np->fldcnt = 0;
- np->flds = 0;
- np->cbuf[np->pos++] = c;
- np->cksum = 0;
- np->msgcksum = -1;
- np->ti = TI_UNK;
- np->state = S_TI_1;
- }
- break;
- case S_TI_1:
- nmea_bufadd(np, c);
- np->state = S_TI_2;
- switch (c) {
- case 'G':
- np->ti = TI_GPS;
- break;
- case 'L':
- np->ti = TI_LORC;
- break;
- default:
- np->state = S_SYNC;
- }
- break;
- case S_TI_2:
- nmea_bufadd(np, c);
- np->state = S_SYNC;
- switch (c) {
- case 'P':
- if (np->ti == TI_GPS)
- np->state = S_MSG;
- break;
- case 'C':
- if (np->ti == TI_LORC)
- np->state = S_MSG;
- break;
- }
- break;
- case S_MSG:
- nmea_bufadd(np, c);
- if (np->pos == 6) {
- np->msg = (np->cbuf[3] << 16) + (np->cbuf[4] << 8) +
- np->cbuf[5];
- switch (np->msg) {
- case MSG_RMC:
- np->flds = 12; /* or 11 */
- np->state = S_DATA;
- break;
- default:
- np->state = S_SYNC;
- }
- }
- break;
- case S_DATA:
- switch (c) {
- case '\n':
- np->cbuf[np->pos] = '\0';
- nmea_decode(np);
- np->state = S_SYNC;
- break;
+ char *fld[MAXFLDS];
+ char *cs;
+ int fldcnt;
+ int cksum, msgcksum;
+ int n;
+
+ fldcnt = 0;
+ cksum = 0;
+
+ /* split into fields and calc checksum */
+ fld[fldcnt++] = &np->cbuf[0]; /* message type */
+ for (cs = NULL, n = 0; n < np->pos && cs == NULL; n++) {
+ switch (np->cbuf[n]) {
case '*':
- np->cbuf[np->pos++] = c;
- np->msgcksum = 0;
- np->state = S_CKSUM;
+ np->cbuf[n] = '\0';
+ cs = &np->cbuf[n + 1];
break;
case ',':
- np->fpos[np->fldcnt++] = np->pos + 1;
- default:
- if (np->pos < NMEAMAX)
- nmea_bufadd(np, c);
- else
- np->state = S_SYNC;
- }
- break;
- case S_CKSUM:
- switch (c) {
- case '\r':
- case '\n':
- np->cbuf[np->pos] = '\0';
- nmea_decode(np);
- np->state = S_SYNC;
+ if (fldcnt < MAXFLDS) {
+ cksum ^= np->cbuf[n];
+ np->cbuf[n] = '\0';
+ fld[fldcnt++] = &np->cbuf[n + 1];
+ } else {
+ DPRINTF(("nr of fields in %s sentence exceeds "
+ "maximum of %d\n", fld[0], MAXFLDS));
+ return;
+ }
break;
default:
- if (np->pos < NMEAMAX && ((c >= '0' && c<= '9') ||
- (c >= 'A' && c <= 'F'))) {
- np->cbuf[np->pos++] = c;
- if (np->msgcksum)
- np->msgcksum <<= 4;
- if (c >= '0' && c<= '9')
- np->msgcksum += c - '0';
- else if (c >= 'A' && c <= 'F')
- np->msgcksum += 10 + c - 'A';
- } else
- np->state = S_SYNC;
+ cksum ^= np->cbuf[n];
}
- break;
}
-}
-/* add a character to the buffer and update the checksum */
-void
-nmea_bufadd(struct nmea *np, int c)
-{
- if (np->pos < NMEAMAX){
- np->cbuf[np->pos++] = c;
- np->cksum ^= c;
+ /* if we have a checksum, verify it */
+ if (cs != NULL) {
+ msgcksum = 0;
+ while (*cs) {
+ if ((*cs >= '0' && *cs <= '9') ||
+ (*cs >= 'A' && *cs <= 'F')) {
+ if (msgcksum)
+ msgcksum <<= 4;
+ if (*cs >= '0' && *cs<= '9')
+ msgcksum += *cs - '0';
+ else if (*cs >= 'A' && *cs <= 'F')
+ msgcksum += 10 + *cs - 'A';
+ cs++;
+ } else {
+ DPRINTF(("bad char %c in checksum\n", *cs));
+ return;
+ }
+ }
+ if (msgcksum != cksum) {
+ DPRINTF(("cksum mismatch"));
+ return;
+ }
}
-}
-void
-nmea_decode(struct nmea *np)
-{
- switch (np->msg) {
- case MSG_RMC:
- nmea_rmc(np);
- break;
- }
+ /* check message type */
+ if (!strcmp(fld[0], "GPRMC"))
+ nmea_gprmc(np, fld, fldcnt);
}
/* Decode the minimum recommended nav info sentence (RMC) */
void
-nmea_rmc(struct nmea *np)
+nmea_gprmc(struct nmea *np, char *fld[], int fldcnt)
{
int64_t date_nano, time_nano, nmea_now;
+#ifdef NMEA_DEBUG
+ int n;
- if (np->fldcnt != 11 && np->fldcnt != 12) {
- DPRINTF(("field count mismatch\n"));
- return;
- }
- if (np->msgcksum >= 0 && np->cksum != np->msgcksum) {
- DPRINTF(("checksum error"));
+ for (n = 0; n < fldcnt; n++)
+ DPRINTF(("%s ", fld[n]));
+ DPRINTF(("\n"));
+#endif
+
+ if (fldcnt != 12 && fldcnt != 13) {
+ DPRINTF(("gprmc: field count mismatch, %d\n", fldcnt));
return;
}
- if (nmea_time_to_nano(&np->cbuf[7], &time_nano)) {
- DPRINTF(("illegal time"));
+ if (nmea_time_to_nano(fld[1], &time_nano)) {
+ DPRINTF(("gprmc: illegal time, %s\n", fld[1]));
return;
}
- if (nmea_date_to_nano(&np->cbuf[np->fpos[8]], &date_nano)) {
- DPRINTF(("illegal date"));
+ if (nmea_date_to_nano(fld[9], &date_nano)) {
+ DPRINTF(("gprmc: illegal date, %s\n", fld[9]));
return;
}
nmea_now = date_nano + time_nano;
if (nmea_now <= np->last) {
- DPRINTF(("time not monotonically increasing\n"));
+ DPRINTF(("gprmc: time not monotonically increasing\n"));
return;
}
np->last = nmea_now;
np->time.value = np->ts.tv_sec * 1000000000LL + np->ts.tv_nsec -
nmea_now;
- np->time.tv.tv_sec = np->tv.tv_sec;
- np->time.tv.tv_usec = np->tv.tv_usec;
+ np->time.tv.tv_sec = np->ts.tv_sec;
+ np->time.tv.tv_usec = np->ts.tv_nsec / 1000L;
if (np->time.status == SENSOR_S_UNKNOWN) {
- strlcpy(np->time.desc, np->ti == TI_GPS ? "GPS" :
- "Loran-C", sizeof(np->time.desc));
- if (np->fldcnt == 12) {
- switch (np->cbuf[np->fpos[11]]) {
+ strlcpy(np->time.desc, "GPS", sizeof(np->time.desc));
+ if (fldcnt == 13) {
+ switch (*fld[12]) {
case 'S':
strlcat(np->time.desc, " simulated",
sizeof(np->time.desc));
@@ -363,7 +285,7 @@ nmea_rmc(struct nmea *np)
np->time.status = SENSOR_S_OK;
np->time.flags &= ~SENSOR_FINVALID;
}
- switch (np->cbuf[np->fpos[1]]) {
+ switch (*fld[2]) {
case 'A':
np->time.status = SENSOR_S_OK;
break;
@@ -371,12 +293,12 @@ nmea_rmc(struct nmea *np)
np->time.status = SENSOR_S_WARN;
break;
default:
- DPRINTF(("unknown warning indication\n"));
+ DPRINTF(("gprmc: unknown warning indication\n"));
}
}
/*
- * convert a NMEA0183 formatted date string to seconds since the epoch
+ * convert a NMEA 0183 formatted date string to seconds since the epoch
* the string must be of the form DDMMYY
* return (0) on success, (-1) if illegal characters are encountered
*/
@@ -391,7 +313,7 @@ nmea_date_to_nano(char *s, int64_t *nano)
/* make sure the input contains only numbers and is six digits long */
for (n = 0, p = s; n < 6 && *p && *p >= '0' && *p <= '9'; n++, p++)
;
- if (n != 6 || (*p != '\0' && *p != ','))
+ if (n != 6 || (*p != '\0'))
return (-1);
year = 2000 + (s[4] - '0') * 10 + (s[5] - '0');
@@ -405,7 +327,7 @@ nmea_date_to_nano(char *s, int64_t *nano)
}
/*
- * convert NMEA0183 formatted time string to nanoseconds since midnight
+ * convert NMEA 0183 formatted time string to nanoseconds since midnight
* the string must be of the form HHMMSS[.[sss]]
* (e.g. 143724 or 143723.615)
* return (0) on success, (-1) if illegal characters are encountered
@@ -458,7 +380,7 @@ nmea_time_to_nano(char *s, int64_t *nano)
}
}
- if (*s != '\0' && *s != ',')
+ if (*s != '\0')
return (-1);
*nano = secs * 1000000000LL + (int64_t)frac * (1000000000 / div);