123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467 |
- /*
- * timehint.c - put time information in SHM segment for ntpd, or to chrony
- *
- * Note that for easy debugging all logging from this file is prefixed
- * with PPS or NTP.
- *
- * This file is Copyright 2010 by the GPSD project
- * SPDX-License-Identifier: BSD-2-clause
- */
- #include "gpsd_config.h" /* must be before all includes */
- #include <errno.h>
- #include <libgen.h>
- #include <math.h>
- #include <stdbool.h>
- #include <string.h>
- #include <sys/socket.h>
- #include <sys/stat.h>
- #include <sys/types.h>
- #include <sys/wait.h>
- #include <time.h> /* for timespec */
- #include <unistd.h>
- #include "timespec.h"
- #include "gpsd.h"
- #include "ntpshm.h"
- /* Note: you can start gpsd as non-root, and have it work with ntpd.
- * However, it will then only use the ntpshm segments 2 3, and higher.
- *
- * Ntpd always runs as root (to be able to control the system clock).
- * After that it often (depending on its host configuration) drops to run as
- * user ntpd and group ntpd.
- *
- * As of February 2015 its rules for the creation of ntpshm segments are:
- *
- * Segments 0 and 1: permissions 0600, i.e. other programs can only
- * read and write as root.
- *
- * Segments 2, 3, and higher:
- * permissions 0666, i.e. other programs can read
- * and write as any user. I.e.: if ntpd has been
- * configured to use these segments, any
- * unprivileged user is allowed to provide data
- * for synchronisation.
- *
- * By default ntpd creates 0 segments (though the documentation is
- * written in such a way as to suggest it creates 4). It can be
- * configured to create up to 217. gpsd creates two segments for each
- * device it can drive; by default this is 8 segments for 4
- * devices,but can be higher if it was compiled with a larger value of
- * MAX_DEVICES.
- *
- * Started as root, gpsd does as ntpd when attaching (creating) the
- * segments. In contrast to ntpd, which only attaches (creates)
- * configured segments, gpsd creates all segments. Thus a gpsd will
- * by default create eight segments 0-7 that an ntpd with default
- * configuration does not watch.
- *
- * Started as non-root, gpsd will only attach (create) segments 2 and
- * above, with permissions 0666. As the permissions are for any user,
- * the creator does not matter.
- *
- * For each GPS module gpsd controls, it will use the attached ntpshm
- * segments in pairs (for coarse clock and pps source, respectively)
- * starting from the first found segments. I.e. started as root, one
- * GPS will deliver data on all segments including 0 and 1; started as
- * non-root, gpsd will be deliver data only on segments 2 and higher.
- *
- * Segments are allocated to activated devices on a first-come-first-served
- * basis. A device's segment is marked unused when the device is closed and
- * may be re-used by devices connected later.
- *
- * To debug, try looking at the live segments this way:
- *
- * ipcs -m
- *
- * results should look like this:
- * ------ Shared Memory Segments --------
- * key shmid owner perms bytes nattch status
- * 0x4e545030 0 root 700 96 2
- * 0x4e545031 32769 root 700 96 2
- * 0x4e545032 163842 root 666 96 1
- * 0x4e545033 196611 root 666 96 1
- * 0x4e545034 253555 root 666 96 1
- * 0x4e545035 367311 root 666 96 1
- *
- * For a bit more data try this:
- * cat /proc/sysvipc/shm
- *
- * If gpsd can not open the segments be sure you are not running SELinux
- * or apparmor.
- *
- * if you see the shared segments (keys 1314148400 -- 1314148405), and
- * no gpsd or ntpd is running, you can remove them like this:
- *
- * ipcrm -M 0x4e545030
- * ipcrm -M 0x4e545031
- * ipcrm -M 0x4e545032
- * ipcrm -M 0x4e545033
- * ipcrm -M 0x4e545034
- * ipcrm -M 0x4e545035
- *
- * Removing these segments is usually not necessary, as the operating system
- * garbage-collects them when they have no attached processes.
- */
- static volatile struct shmTime *getShmTime(struct gps_context_t *context,
- int unit)
- {
- int shmid;
- unsigned int perms;
- volatile struct shmTime *p;
- // set the SHM perms the way ntpd does
- if (unit < 2) {
- // we are root, be careful
- perms = 0600;
- } else {
- // we are not root, try to work anyway
- perms = 0666;
- }
- /*
- * Note: this call requires root under BSD, and possibly on
- * well-secured Linux systems. This is why ntpshm_context_init() has to be
- * called before privilege-dropping.
- */
- shmid = shmget((key_t) (NTPD_BASE + unit),
- sizeof(struct shmTime), (int)(IPC_CREAT | perms));
- if (shmid == -1) {
- GPSD_LOG(LOG_ERROR, &context->errout,
- "NTP: shmget(%ld, %zd, %o) fail: %s\n",
- (long int)(NTPD_BASE + unit), sizeof(struct shmTime),
- (int)perms, strerror(errno));
- return NULL;
- }
- p = (struct shmTime *)shmat(shmid, 0, 0);
- if ((int)(long)p == -1) {
- GPSD_LOG(LOG_ERROR, &context->errout,
- "NTP: shmat failed: %s\n",
- strerror(errno));
- return NULL;
- }
- GPSD_LOG(LOG_PROG, &context->errout,
- "NTP: shmat(%d,0,0) succeeded, segment %d\n",
- shmid, unit);
- return p;
- }
- void ntpshm_context_init(struct gps_context_t *context)
- /* Attach all NTP SHM segments. Called once at startup, while still root. */
- {
- int i;
- for (i = 0; i < NTPSHMSEGS; i++) {
- // Only grab the first two when running as root.
- if (2 <= i || 0 == getuid()) {
- context->shmTime[i] = getShmTime(context, i);
- }
- }
- memset(context->shmTimeInuse, 0, sizeof(context->shmTimeInuse));
- }
- static volatile struct shmTime *ntpshm_alloc(struct gps_context_t *context)
- /* allocate NTP SHM segment. return its segment number, or -1 */
- {
- int i;
- for (i = 0; i < NTPSHMSEGS; i++)
- if (context->shmTime[i] != NULL && !context->shmTimeInuse[i]) {
- context->shmTimeInuse[i] = true;
- /*
- * In case this segment gets sent to ntpd before an
- * ephemeris is available, the LEAP_NOTINSYNC value will
- * tell ntpd that this source is in a "clock alarm" state
- * and should be ignored. The goal is to prevent ntpd
- * from declaring the GPS a falseticker before it gets
- * all its marbles together.
- */
- memset((void *)context->shmTime[i], 0, sizeof(struct shmTime));
- context->shmTime[i]->mode = 1;
- context->shmTime[i]->leap = LEAP_NOTINSYNC;
- context->shmTime[i]->precision = -20;/* initially 1 micro sec */
- context->shmTime[i]->nsamples = 3; /* stages of median filter */
- return context->shmTime[i];
- }
- return NULL;
- }
- static bool ntpshm_free(struct gps_context_t * context,
- volatile struct shmTime *s)
- /* free NTP SHM segment */
- {
- int i;
- for (i = 0; i < NTPSHMSEGS; i++)
- if (s == context->shmTime[i]) {
- context->shmTimeInuse[i] = false;
- return true;
- }
- return false;
- }
- void ntpshm_session_init(struct gps_device_t *session)
- {
- /* mark NTPD shared memory segments as unused */
- session->shm_clock = NULL;
- session->shm_pps = NULL;
- }
- /* put a received fix time into shared memory for NTP */
- int ntpshm_put(struct gps_device_t *session, volatile struct shmTime *shmseg,
- struct timedelta_t *td)
- {
- char real_str[TIMESPEC_LEN];
- char clock_str[TIMESPEC_LEN];
- /* Any NMEA will be about -1 or -2. Garmin GPS-18/USB is around -6 or -7. */
- int precision = -20; /* default precision, 1 micro sec */
- if (shmseg == NULL) {
- GPSD_LOG(LOG_RAW, &session->context->errout, "NTP:PPS: missing shm\n");
- return 0;
- }
- // FIXME: make NMEA precision -1
- if (shmseg == session->shm_pps) {
- /* precision is a floor so do not make it tight */
- if ( source_usb == session->sourcetype ) {
- /* if PPS over USB, then precision = -10, 1 milli sec */
- precision = -10;
- } else {
- /* likely PPS over serial, precision = -20, 1 micro sec */
- precision = -20;
- }
- }
- ntp_write(shmseg, td, precision, session->context->leap_notify);
- GPSD_LOG(LOG_PROG, &session->context->errout,
- "NTP: ntpshm_put(%s,%d) %s @ %s\n",
- session->gpsdata.dev.path,
- precision,
- timespec_str(&td->real, real_str, sizeof(real_str)),
- timespec_str(&td->clock, clock_str, sizeof(clock_str)));
- return 1;
- }
- #define SOCK_MAGIC 0x534f434b
- struct sock_sample {
- struct timeval tv;
- double offset;
- int pulse;
- int leap; /* notify that a leap second is upcoming */
- // cppcheck-suppress unusedStructMember
- int _pad;
- int magic; /* must be SOCK_MAGIC */
- };
- static void init_hook(struct gps_device_t *session)
- /* for chrony SOCK interface, which allows nSec timekeeping */
- {
- /* open the chrony socket */
- char chrony_path[GPS_PATH_MAX];
- session->chronyfd = -1;
- if ( 0 == getuid() ) {
- /* this case will fire on command-line devices;
- * they're opened before priv-dropping. Matters because
- * usually only root can use /run or /var/run.
- */
- (void)snprintf(chrony_path, sizeof (chrony_path),
- RUNDIR "/chrony.%s.sock", basename(session->gpsdata.dev.path));
- } else {
- (void)snprintf(chrony_path, sizeof (chrony_path),
- "/tmp/chrony.%s.sock", basename(session->gpsdata.dev.path));
- }
- if (access(chrony_path, F_OK) != 0) {
- GPSD_LOG(LOG_PROG, &session->context->errout,
- "PPS:%s chrony socket %s doesn't exist\n",
- session->gpsdata.dev.path, chrony_path);
- } else {
- session->chronyfd = netlib_localsocket(chrony_path, SOCK_DGRAM);
- if (session->chronyfd < 0)
- GPSD_LOG(LOG_PROG, &session->context->errout,
- "PPS:%s connect chrony socket failed: %s, error: %d, "
- "errno: %d/%s\n",
- session->gpsdata.dev.path,
- chrony_path, session->chronyfd, errno, strerror(errno));
- else
- GPSD_LOG(LOG_RAW, &session->context->errout,
- "PPS:%s using chrony socket: %s\n",
- session->gpsdata.dev.path, chrony_path);
- }
- }
- /* td is the real time and clock time of the edge */
- /* offset is actual_ts - clock_ts */
- static void chrony_send(struct gps_device_t *session, struct timedelta_t *td)
- {
- char real_str[TIMESPEC_LEN];
- char clock_str[TIMESPEC_LEN];
- struct sock_sample sample;
- struct tm tm;
- int leap_notify = session->context->leap_notify;
- /*
- * insist that leap seconds only happen in june and december
- * GPS emits leap pending for 3 months prior to insertion
- * NTP expects leap pending for only 1 month prior to insertion
- * Per http://bugs.ntp.org/1090
- *
- * ITU-R TF.460-6, Section 2.1, says leap seconds can be primarily
- * in Jun/Dec but may be in March or September
- */
- (void)gmtime_r( &(td->real.tv_sec), &tm);
- if ( 5 != tm.tm_mon && 11 != tm.tm_mon ) {
- /* Not june, not December, no way */
- leap_notify = LEAP_NOWARNING;
- }
- /* chrony expects tv-sec since Jan 1970 */
- sample.pulse = 0;
- sample.leap = leap_notify;
- sample.magic = SOCK_MAGIC;
- /* chronyd wants a timeval, not a timspec, not to worry, it is
- * just the top of the second */
- TSTOTV(&sample.tv, &td->clock);
- /* calculate the offset as a timespec to not lose precision */
- /* if tv_sec greater than 2 then tv_nsec loses precision, but
- * not a big deal as slewing will be required */
- sample.offset = TS_SUB_D(&td->real, &td->clock);
- sample._pad = 0;
- GPSD_LOG(LOG_RAW, &session->context->errout,
- "PPS chrony_send %s @ %s Offset: %0.9f\n",
- timespec_str(&td->real, real_str, sizeof(real_str)),
- timespec_str(&td->clock, clock_str, sizeof(clock_str)),
- sample.offset);
- (void)send(session->chronyfd, &sample, sizeof (sample), 0);
- }
- static char *report_hook(volatile struct pps_thread_t *pps_thread,
- struct timedelta_t *td)
- /* ship the time of a PPS event to ntpd and/or chrony */
- {
- char *log1;
- struct gps_device_t *session = (struct gps_device_t *)pps_thread->context;
- /* PPS only source never get any serial info
- * so no NTPTIME_IS or fixcnt */
- if ( source_pps != session->sourcetype) {
- /* FIXME! these two validations need to move back into ppsthread.c */
- if ( !session->ship_to_ntpd)
- return "skipped ship_to_ntp=0";
- /*
- * Only listen to PPS after several consecutive fixes,
- * otherwise time may be inaccurate. We know this is
- * required on all Garmin and u-blox. Safest to do it
- * for all cases as we have no other general way to know
- * if PPS is good.
- * Allow override with batteryRTC to allow foot shots.
- */
- if (false == session->context->batteryRTC &&
- session->fixcnt <= NTP_MIN_FIXES &&
- (session->gpsdata.set & GOODTIME_IS) == 0)
- return "no fix";
- }
- /* FIXME? how to log socket AND shm reported? */
- log1 = "accepted";
- if ( 0 <= session->chronyfd ) {
- log1 = "accepted chrony sock";
- chrony_send(session, td);
- }
- if (session->shm_pps != NULL)
- (void)ntpshm_put(session, session->shm_pps, td);
- /* session context might have a hook set, too */
- if (session->context->pps_hook != NULL)
- session->context->pps_hook(session, td);
- return log1;
- }
- void ntpshm_link_deactivate(struct gps_device_t *session)
- /* release ntpshm storage for a session */
- {
- if (session->shm_clock != NULL) {
- (void)ntpshm_free(session->context, session->shm_clock);
- session->shm_clock = NULL;
- }
- if (session->shm_pps != NULL) {
- pps_thread_deactivate(&session->pps_thread);
- if (session->chronyfd != -1)
- (void)close(session->chronyfd);
- (void)ntpshm_free(session->context, session->shm_pps);
- session->shm_pps = NULL;
- }
- }
- void ntpshm_link_activate(struct gps_device_t *session)
- /* set up ntpshm storage for a session */
- {
- /* don't talk to NTP when we're running inside the test harness */
- if (session->sourcetype == source_pty)
- return;
- if (session->sourcetype != source_pps ) {
- /* allocate a shared-memory segment for "NMEA" time data */
- session->shm_clock = ntpshm_alloc(session->context);
- if (session->shm_clock == NULL) {
- GPSD_LOG(LOG_WARN, &session->context->errout,
- "NTP: ntpshm_alloc() failed\n");
- return;
- }
- }
- if (session->sourcetype == source_usb
- || session->sourcetype == source_rs232
- || session->sourcetype == source_pps) {
- /* We also have the 1pps capability, allocate a shared-memory segment
- * for the 1pps time data and launch a thread to capture the 1pps
- * transitions
- */
- if ((session->shm_pps = ntpshm_alloc(session->context)) == NULL) {
- GPSD_LOG(LOG_WARN, &session->context->errout,
- "PPS: ntpshm_alloc(1) failed\n");
- } else {
- init_hook(session);
- session->pps_thread.report_hook = report_hook;
- #ifdef MAGIC_HAT_ENABLE
- /*
- * The HAT kludge. If we're using the HAT GPS on a
- * Raspberry Pi or a workalike like the ODROIDC2, and
- * there is a static "first PPS", and we have access because
- * we're root, assume we want to use KPPS.
- */
- if (strcmp(session->pps_thread.devicename, MAGIC_HAT_GPS) == 0
- || strcmp(session->pps_thread.devicename,
- MAGIC_LINK_GPS) == 0) {
- char *first_pps = pps_get_first();
- if (access(first_pps, R_OK | W_OK) == 0)
- session->pps_thread.devicename = first_pps;
- }
- #endif /* MAGIC_HAT_ENABLE */
- pps_thread_activate(&session->pps_thread);
- }
- }
- }
- /* end */
- // vim: set expandtab shiftwidth=4
|