summaryrefslogblamecommitdiffstats
path: root/core/timer.c
blob: 14ccd46664c4d2b9136a2f773627efecc9e6ec99 (plain) (tree)































































































































































                                                                                                 
/*
 * libdivecomputer
 *
 * Copyright (C) 2018 Jef Driesen
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2.1 of the License, or (at your option) any later version.
 *
 * This library is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this library; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
 * MA 02110-1301 USA
 */

#ifdef HAVE_CONFIG_H
#include "config.h"
#endif

#include <stdlib.h>

#ifdef _WIN32
#define NOGDI
#include <windows.h>
#else
#include <time.h>
#include <sys/time.h>
#ifdef HAVE_MACH_MACH_TIME_H
#include <mach/mach_time.h>
#endif
#endif

#include "timer.h"

struct dc_timer_t {
#if defined (_WIN32)
	LARGE_INTEGER timestamp;
	LARGE_INTEGER frequency;
#elif defined (HAVE_CLOCK_GETTIME)
	struct timespec timestamp;
#elif defined (HAVE_MACH_ABSOLUTE_TIME)
	uint64_t timestamp;
	mach_timebase_info_data_t info;
#else
	struct timeval timestamp;
#endif
};

dc_status_t
dc_timer_new (dc_timer_t **out)
{
	dc_timer_t *timer = NULL;

	if (out == NULL)
		return DC_STATUS_INVALIDARGS;

	timer = (dc_timer_t *) malloc (sizeof (dc_timer_t));
	if (timer == NULL) {
		return DC_STATUS_NOMEMORY;
	}

#if defined (_WIN32)
	if (!QueryPerformanceFrequency(&timer->frequency) ||
		!QueryPerformanceCounter(&timer->timestamp)) {
		free(timer);
		return DC_STATUS_IO;
	}
#elif defined (HAVE_CLOCK_GETTIME)
	if (clock_gettime(CLOCK_MONOTONIC, &timer->timestamp) != 0) {
		free(timer);
		return DC_STATUS_IO;
	}
#elif defined (HAVE_MACH_ABSOLUTE_TIME)
	if (mach_timebase_info(&timer->info) != KERN_SUCCESS) {
		free(timer);
		return DC_STATUS_IO;
	}

	timer->timestamp = mach_absolute_time();
#else
	if (gettimeofday (&timer->timestamp, NULL) != 0) {
		free(timer);
		return DC_STATUS_IO;
	}
#endif

	*out = timer;

	return DC_STATUS_SUCCESS;
}

dc_status_t
dc_timer_now (dc_timer_t *timer, dc_usecs_t *usecs)
{
	dc_status_t status = DC_STATUS_SUCCESS;
	dc_usecs_t value = 0;

	if (timer == NULL) {
		status = DC_STATUS_INVALIDARGS;
		goto out;
	}

#if defined (_WIN32)
	LARGE_INTEGER now;
	if (!QueryPerformanceCounter(&now)) {
		status = DC_STATUS_IO;
		goto out;
	}

	value = (now.QuadPart - timer->timestamp.QuadPart) * 1000000 / timer->frequency.QuadPart;
#elif defined (HAVE_CLOCK_GETTIME)
	struct timespec now, delta;
	if (clock_gettime(CLOCK_MONOTONIC, &now) != 0) {
		status = DC_STATUS_IO;
		goto out;
	}

	if (now.tv_nsec < timer->timestamp.tv_nsec) {
		delta.tv_nsec = 1000000000 + now.tv_nsec - timer->timestamp.tv_nsec;
		delta.tv_sec = now.tv_sec - timer->timestamp.tv_sec - 1;
	} else {
		delta.tv_nsec = now.tv_nsec - timer->timestamp.tv_nsec;
		delta.tv_sec = now.tv_sec - timer->timestamp.tv_sec;
	}

	value = (dc_usecs_t) delta.tv_sec * 1000000 + delta.tv_nsec / 1000;
#elif defined (HAVE_MACH_ABSOLUTE_TIME)
	uint64_t now = mach_absolute_time();
	value = (now - timer->timestamp) * timer->info.numer / (timer->info.denom * 1000);
#else
	struct timeval now, delta;
	if (gettimeofday (&now, NULL) != 0) {
		status = DC_STATUS_IO;
		goto out;
	}

	timersub (&now, &timer->timestamp, &delta);

	value = (dc_usecs_t) delta.tv_sec * 1000000 + delta.tv_usec;
#endif

out:
	if (usecs)
		*usecs = value;

	return status;
}

dc_status_t
dc_timer_free (dc_timer_t *timer)
{
	free (timer);

	return DC_STATUS_SUCCESS;
}