/*
 * ntp_ppsdev.c - PPS-device support
 *
 * Written by Juergen Perlinger (perlinger@ntp.org) for the NTP project.
 * The contents of 'html/copyright.html' apply.
 * ---------------------------------------------------------------------
 * Helper code to work around (or with) a Linux 'specialty': PPS devices
 * are created via attaching the PPS line discipline to a TTY.  This
 * creates new pps devices, and the PPS API is *not* available through
 * the original TTY fd.
 *
 * Findig the PPS device associated with a TTY is possible but needs
 * quite a bit of file system traversal & lookup in the 'sysfs' tree.
 *
 * The code below does the job for kernel versions 4 & 5, and will
 * probably work for older and newer kernels, too... and in any case, if
 * the device or symlink to the PPS device with the given name exists,
 * it will take precedence anyway.
 * ---------------------------------------------------------------------
 */
#ifdef __linux__
# define _GNU_SOURCE
#endif

#include "config.h"

#include "ntpd.h"

#ifdef REFCLOCK

#if defined(HAVE_UNISTD_H)
# include <unistd.h>
#endif
#if defined(HAVE_FCNTL_H)
# include <fcntl.h>
#endif

#include <stdlib.h>

/* -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=- */
#if defined(__linux__) && defined(HAVE_OPENAT) && defined(HAVE_FDOPENDIR)
#define WITH_PPSDEV_MATCH
/* -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=- */

#include <stdio.h>
#include <dirent.h>
#include <string.h>
#include <errno.h>

#include <sys/ioctl.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/sysmacros.h>
#include <linux/tty.h>

typedef int BOOL;
#ifndef TRUE
# define TRUE 1
#endif
#ifndef FALSE
# define FALSE 0
#endif

static const int OModeF = O_CLOEXEC|O_RDONLY|O_NOCTTY;
static const int OModeD = O_CLOEXEC|O_RDONLY|O_DIRECTORY;

/* ------------------------------------------------------------------ */
/* extended directory stream
 */
typedef struct {
	int  dfd;	/* file descriptor for dir for 'openat()' */
	DIR *dir;	/* directory stream for iteration         */
} XDIR;

static void
xdirClose(
	XDIR *pxdir)
{
	if (NULL != pxdir->dir)
		closedir(pxdir->dir); /* closes the internal FD, too! */
	else if (-1 != pxdir->dfd)
		close(pxdir->dfd);    /* otherwise _we_ have to do it */
	pxdir->dfd = -1;
	pxdir->dir = NULL;
}

static BOOL
xdirOpenAt(
	XDIR       *pxdir,
	int         fdo  ,
	const char *path )
{
	/* Officially, the directory stream owns the file discriptor it
	 * received via 'fdopendir()'.  But for the purpose of 'openat()'
	 * it's ok to keep the value around -- even if we should do
	 * _absolutely_nothing_ with it apart from using it as a path
	 * reference!
	 */
	pxdir->dir = NULL;
	if (-1 == (pxdir->dfd = openat(fdo, path, OModeD)))
		goto fail;
	if (NULL == (pxdir->dir = fdopendir(pxdir->dfd)))
		goto fail;
	return TRUE;
	
  fail:
	xdirClose(pxdir);
	return FALSE;
}

/* --------------------------------------------------------------------
 * read content of a file (with a size limit) into a piece of allocated
 * memory and trim any trailing whitespace.
 *
 * The issue here is that several files in the 'sysfs' tree claim a size
 * of 4096 bytes when you 'stat' them -- but reading gives EOF after a
 * few chars.  (I *can* understand why the kernel takes this shortcut.
 * it's just a bit unwieldy...)
 */
static char*
readFileAt(
	int         rfd ,
	const char *path)
{
	struct stat sb;
	char *ret = NULL;
	ssize_t rdlen;
	int dfd;
	
	if (-1 == (dfd = openat(rfd, path, OModeF)) || -1 == fstat(dfd, &sb))
		goto fail;
	if ((sb.st_size > 0x2000) || (NULL == (ret = malloc(sb.st_size + 1))))
		goto fail;
	if (1 > (rdlen = read(dfd, ret, sb.st_size)))
		goto fail;
	close(dfd);

	while (rdlen > 0 && ret[rdlen - 1] <= ' ')
		--rdlen;
	ret[rdlen] = '\0';
	return ret;

  fail:
	free(ret);
	if (-1 != dfd)
		close(dfd);
	return NULL;    
}

/* --------------------------------------------------------------------
 * Scan the "/dev" directory for a device with a given major and minor 
 * device id. Return the path if found.
 */
static char*
findDevByDevId(
	dev_t rdev)
{
	struct stat    sb;
	struct dirent *dent;
	XDIR           xdir;
	char          *name = NULL;
	
	if (!xdirOpenAt(&xdir, AT_FDCWD, "/dev"))
		goto done;
	
	while (!name && (dent = readdir(xdir.dir))) {
		if (-1 == fstatat(xdir.dfd, dent->d_name,
				  &sb, AT_SYMLINK_NOFOLLOW))
			continue;
		if (!S_ISCHR(sb.st_mode))
			continue;
		if (sb.st_rdev == rdev) {
			if (-1 == asprintf(&name, "/dev/%s", dent->d_name))
				name = NULL;
		}
	}
	xdirClose(&xdir);

  done:
	return name;
}

/* --------------------------------------------------------------------
 * Get the mofor:minor device id for a character device file descriptor
 */
static BOOL
getCharDevId(
	int          fd ,
	dev_t       *out,
	struct stat *psb)
{
	BOOL        rc = FALSE;
	struct stat sb;
	
	if (NULL == psb)
		psb = &sb;
	if (-1 != fstat(fd, psb)) {
		rc = S_ISCHR(psb->st_mode);
		if (rc)
			*out = psb->st_rdev;
		else
			errno = EINVAL;
	}
	return rc;
}

/* --------------------------------------------------------------------
 * given the dir-fd of a pps instance dir in the linux sysfs tree, get
 * the device IDs for the PPS device and the associated TTY.
 */
static BOOL
getPpsTuple(
	int   fdDir,
	dev_t *pTty,
	dev_t *pPps)
{
	BOOL          rc = FALSE;
	unsigned long dmaj, dmin;
	struct stat   sb;
	char         *bufp, *endp, *scan;

	/* 'path' contains the primary path to the associated TTY:
	 * we 'stat()' for the device id in 'st_rdev'.
	 */
	if (NULL == (bufp = readFileAt(fdDir, "path")))
		goto done;
	if ((-1 == stat(bufp, &sb)) || !S_ISCHR(sb.st_mode))
		goto done;
	*pTty = sb.st_rdev;
	free(bufp);

	/* 'dev' holds the device ID of the PPS device as 'major:minor'
	 * in text format.   *sigh* couldn't that simply be the name of
	 * the PPS device itself, as in 'path' above??? But nooooo....
	 */
	if (NULL == (bufp = readFileAt(fdDir, "dev")))
		goto done;
	dmaj = strtoul((scan = bufp), &endp, 10);
	if ((endp == scan) || (*endp != ':') || (dmaj >= 256))
		goto done;
	dmin = strtoul((scan = endp + 1), &endp, 10);
	if ((endp == scan) || (*endp >= ' ') || (dmin >= 256))
		goto done;
	*pPps = makedev((unsigned int)dmaj, (unsigned int)dmin);
	rc = TRUE;
	
  done:
	free(bufp);
	return rc;	
}

/* --------------------------------------------------------------------
 * for a given (TTY) device id, lookup the corresponding PPS device id
 * by processing the contents of the kernel sysfs tree.
 * Returns false if no such PS device can be found; otherwise set the
 * ouput parameter to the PPS dev id and return true...
 */
static BOOL
findPpsDevId(
	dev_t  ttyId ,
	dev_t *pPpsId)
{
	BOOL           found = FALSE;
	XDIR           ClassDir;
	struct dirent *dent;
	dev_t          othId, ppsId;
	int            fdDevDir;
	
	if (!xdirOpenAt(&ClassDir, AT_FDCWD, "/sys/class/pps"))
		goto done;
		
	while (!found && (dent = readdir(ClassDir.dir))) {

		/* If the entry is not a referring to a PPS device or
		 * if we can't open the directory for reading, skipt it:
		 */
		if (strncmp("pps", dent->d_name, 3))
			continue;
		fdDevDir = openat(ClassDir.dfd, dent->d_name, OModeD);
		if (-1 == fdDevDir)
			continue;

		/* get the data and check if device ID for the TTY
		 * is what we're looking for:
		 */
		found = getPpsTuple(fdDevDir, &othId, &ppsId)
		    && (ttyId == othId);
		close(fdDevDir);
	}
	
	xdirClose(&ClassDir);
	
	if (found)
		*pPpsId = ppsId;
  done:
	return found;
}

/* --------------------------------------------------------------------
 * Return the path to a PPS device related to tghe TT fd given. The
 * function might even try to instantiate such a PPS device when
 * running es effective root.  Returns NULL if no PPS device can be
 * established; otherwise it is a 'malloc()'ed area that should be
 * 'free()'d after use.
 */
static char*
findMatchingPpsDev(
	int fdtty)
{
	struct stat sb;
	dev_t       ttyId, ppsId;
	int         fdpps, ldisc = N_PPS;
	char       *dpath = NULL;

	/* Without the device identifier of the TTY, we're busted: */
	if (!getCharDevId(fdtty, &ttyId, &sb))
		goto done;

	/* If we find a matching PPS device ID, return the path to the
	 * device. It might not open, but it's the best we can get.
	 */
	if (findPpsDevId(ttyId, &ppsId)) {
		dpath = findDevByDevId(ppsId);
		goto done;
	}
	
#   ifdef ENABLE_MAGICPPS
	/* 'magic' PPS support -- try to instantiate missing PPS devices
	 * on-the-fly.  Our mileage may vary -- running as root at that
	 * moment is vital for success.  (We *can* create the PPS device
	 * as ordnary user, but we won't be able to open it!)
	 */
	
	/* If we're root, try to push the PPS LDISC to the tty FD. If
	 * that does not work out, we're busted again:
	 */
	if ((0 != geteuid()) || (-1 == ioctl(fdtty, TIOCSETD, &ldisc)))
		goto done;
	msyslog(LOG_INFO, "auto-instantiated PPS device for device %u:%u",
		major(ttyId), minor(ttyId));

	/* We really should find a matching PPS device now. And since
	 * we're root (see above!), we should be able to open that device.
	 */
	if (findPpsDevId(ttyId, &ppsId))
		dpath = findDevByDevId(ppsId);
	if (!dpath)
		goto done;

	/* And since we're 'root', we might as well try to clone the
	 * ownership and access rights from the original TTY to the
	 * PPS device.  If that does not work, we just have to live with
	 * what we've got so far...
	 */
	if (-1 == (fdpps = open(dpath, OModeF))) {
		msyslog(LOG_ERR, "could not open auto-created '%s': %m", dpath);
		goto done;
	}
	if (-1 == fchmod(fdpps, sb.st_mode)) {
		msyslog(LOG_ERR, "could not chmod auto-created '%s': %m", dpath);
	}
	if (-1 == fchown(fdpps, sb.st_uid, sb.st_gid)) {
		msyslog(LOG_ERR, "could not chown auto-created '%s': %m", dpath);
	}
	close(fdpps);
#   else
	(void)ldisc;
#   endif
	
  done:
	/* Whatever we go so far, that's it. */
	return dpath;
}

/* -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=- */
#endif /* linux PPS device matcher */
/* -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=- */

#include "ntp_clockdev.h"

int
ppsdev_reopen(
	const sockaddr_u *srcadr,
	int         ttyfd  , /* current tty FD, or -1 */
	int         ppsfd  , /* current pps FD, or -1 */
	const char *ppspath, /* path to pps device, or NULL */
	int         omode  , /* open mode for pps device */
	int         oflags ) /* openn flags for pps device */
{
	int retfd = -1;
	const char *altpath;

	/* avoid 'unused' warnings: we might not use all args, no
	 * thanks to conditional compiling:)
	 */
	(void)ppspath;
	(void)omode;
	(void)oflags;

	if (NULL != (altpath = clockdev_lookup(srcadr, 1)))
		ppspath = altpath;

#   if defined(__unix__) && !defined(_WIN32)
	if (-1 == retfd) {	
		if (ppspath && *ppspath) {
			retfd = open(ppspath, omode, oflags);
			msyslog(LOG_INFO, "ppsdev_open(%s) %s",
				ppspath, (retfd != -1 ? "succeeded" : "failed"));
		}
	}
#   endif
	
#   if defined(WITH_PPSDEV_MATCH)
	if ((-1 == retfd) && (-1 != ttyfd)) {	
		char *xpath = findMatchingPpsDev(ttyfd);
		if (xpath && *xpath) {
			retfd = open(xpath, omode, oflags);
			msyslog(LOG_INFO, "ppsdev_open(%s) %s",
				xpath, (retfd != -1 ? "succeeded" : "failed"));
		}
		free(xpath);
	}
#   endif
	
	/* BSDs and probably SOLARIS can use the TTY fd for the PPS API,
	 * and so does Windows where the PPS API is implemented via an
	 * IOCTL.  Likewise does the 'SoftPPS' implementation in Windows
	 * based on COM Events.  So, if everything else fails, simply
	 * try the FD given for the TTY/COMport...
	 */
	if (-1 == retfd)
		retfd = ppsfd;
	if (-1 == retfd)
		retfd = ttyfd;

	/* Close the old pps FD, but only if the new pps FD is neither
	 * the tty FD nor the existing pps FD!
	 */
	if ((retfd != ttyfd) && (retfd != ppsfd))
		ppsdev_close(ttyfd, ppsfd);
	
	return retfd;
}

void
ppsdev_close(
	int ttyfd, /* current tty FD, or -1 */
	int ppsfd) /* current pps FD, or -1 */
{
	/* The pps fd might be the same as the tty fd.  We close the pps
	 * channel only if it's valid and _NOT_ the tty itself:
	 */
	if ((-1 != ppsfd) && (ttyfd != ppsfd))
		close(ppsfd);
}
/* --*-- that's all folks --*-- */
#else
NONEMPTY_TRANSLATION_UNIT
#endif /* !defined(REFCLOCK) */