1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101
|
/* $Id: dgnss.c 3771 2006-11-02 05:15:20Z esr $ */
/* dgnss.c -- common interface to a number of Differential GNSS services */
#include <sys/types.h>
#include <sys/socket.h>
#include <string.h>
#include <unistd.h>
#include <errno.h>
#include "gpsd_config.h"
#include "gpsd.h"
#define DGNSS_PROTO_DGPSIP "dgpsip://"
#define DGNSS_PROTO_NTRIP "ntrip://"
/* Where to find the list of DGPSIP correction servers, if there is one */
#define DGPSIP_SERVER_LIST "/usr/share/gpsd/dgpsip-servers"
bool dgnss_url(char *name)
/* is given string a valid URL for DGPS service? */
{
return
strncmp(name,DGNSS_PROTO_NTRIP,strlen(DGNSS_PROTO_NTRIP))==0
|| strncmp(name,DGNSS_PROTO_DGPSIP,strlen(DGNSS_PROTO_DGPSIP))==0;
}
/*@ -branchstate */
int dgnss_open(struct gps_context_t *context, char *dgnss_service)
/* open a connection to a DGNSS service */
{
#ifdef NTRIP_ENABLE
if (strncmp(dgnss_service,DGNSS_PROTO_NTRIP,strlen(DGNSS_PROTO_NTRIP))==0)
return ntrip_open(context, dgnss_service + strlen(DGNSS_PROTO_NTRIP));
#endif
if (strncmp(dgnss_service,DGNSS_PROTO_DGPSIP,strlen(DGNSS_PROTO_DGPSIP))==0)
return dgpsip_open(context, dgnss_service + strlen(DGNSS_PROTO_DGPSIP));
#ifndef REQUIRE_DGNSS_PROTO
return dgpsip_open(context, dgnss_service);
#else
gpsd_report(LOG_ERROR, "Unknown or unspecified DGNSS protocol for service %s\n",
dgnss_service);
return -1;
#endif
}
/*@ +branchstate */
int dgnss_poll(struct gps_context_t *context)
/* poll the DGNSS service for a correction report */
{
if (context->dsock > -1) {
context->rtcmbytes = read(context->dsock, context->rtcmbuf, sizeof(context->rtcmbuf));
if ((context->rtcmbytes == -1 && errno != EAGAIN) ||
(context->rtcmbytes == 0)) {
(void)shutdown(context->dsock, SHUT_RDWR);
(void)close(context->dsock);
return -1;
} else
context->rtcmtime = timestamp();
}
return 0;
}
void dgnss_report(struct gps_device_t *session)
/* may be time to ship a usage report to the DGNSS service */
{
if (session->context->dgnss_service == dgnss_dgpsip)
dgpsip_report(session);
#ifdef NTRIP_ENABLE
else if (session->context->dgnss_service == dgnss_ntrip)
ntrip_report(session);
#endif
}
void dgnss_autoconnect(struct gps_context_t *context, double lat, double lon)
{
if (context->dgnss_service != dgnss_ntrip) {
dgpsip_autoconnect(context, lat, lon, DGPSIP_SERVER_LIST);
}
}
void rtcm_relay(struct gps_device_t *session)
/* pass a DGNSS connection report to a session */
{
if (session->gpsdata.gps_fd !=-1
&& session->context->rtcmbytes > -1
&& session->rtcmtime < session->context->rtcmtime
&& session->device_type->rtcm_writer != NULL) {
if (session->device_type->rtcm_writer(session,
session->context->rtcmbuf,
(size_t)session->context->rtcmbytes) == 0)
gpsd_report(LOG_ERROR, "Write to rtcm sink failed\n");
else {
session->rtcmtime = timestamp();
gpsd_report(LOG_IO, "<= DGPS: %d bytes of RTCM relayed.\n", session->context->rtcmbytes);
}
}
}
|