2021-07-14 00:54:39 +02:00
|
|
|
#include "fsfw/timemanager/Clock.h"
|
2018-07-13 15:56:37 +02:00
|
|
|
|
2021-07-14 00:54:39 +02:00
|
|
|
#include <cstdlib>
|
|
|
|
#include <ctime>
|
2018-07-13 15:56:37 +02:00
|
|
|
|
2022-02-02 10:29:30 +01:00
|
|
|
#include "FreeRTOS.h"
|
|
|
|
#include "fsfw/globalfunctions/timevalOperations.h"
|
|
|
|
#include "fsfw/osal/freertos/Timekeeper.h"
|
|
|
|
#include "task.h"
|
|
|
|
|
|
|
|
// TODO sanitize input?
|
|
|
|
// TODO much of this code can be reused for tick-only systems
|
2018-07-13 15:56:37 +02:00
|
|
|
|
2022-02-02 10:29:30 +01:00
|
|
|
uint32_t Clock::getTicksPerSecond(void) { return 1000; }
|
2018-07-13 15:56:37 +02:00
|
|
|
|
|
|
|
ReturnValue_t Clock::setClock(const TimeOfDay_t* time) {
|
2022-02-02 10:29:30 +01:00
|
|
|
timeval time_timeval;
|
2018-07-13 15:56:37 +02:00
|
|
|
|
2022-02-02 10:29:30 +01:00
|
|
|
ReturnValue_t result = convertTimeOfDayToTimeval(time, &time_timeval);
|
|
|
|
if (result != HasReturnvaluesIF::RETURN_OK) {
|
|
|
|
return result;
|
|
|
|
}
|
2018-07-13 15:56:37 +02:00
|
|
|
|
2022-02-02 10:29:30 +01:00
|
|
|
return setClock(&time_timeval);
|
2018-07-13 15:56:37 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
ReturnValue_t Clock::setClock(const timeval* time) {
|
2022-02-02 10:29:30 +01:00
|
|
|
timeval uptime = getUptime();
|
2018-07-13 15:56:37 +02:00
|
|
|
|
2022-02-02 10:29:30 +01:00
|
|
|
timeval offset = *time - uptime;
|
2018-07-13 15:56:37 +02:00
|
|
|
|
2022-02-02 10:29:30 +01:00
|
|
|
Timekeeper::instance()->setOffset(offset);
|
2018-07-13 15:56:37 +02:00
|
|
|
|
2022-02-02 10:29:30 +01:00
|
|
|
return HasReturnvaluesIF::RETURN_OK;
|
2018-07-13 15:56:37 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
ReturnValue_t Clock::getClock_timeval(timeval* time) {
|
2022-02-02 10:29:30 +01:00
|
|
|
timeval uptime = getUptime();
|
2018-07-13 15:56:37 +02:00
|
|
|
|
2022-02-02 10:29:30 +01:00
|
|
|
timeval offset = Timekeeper::instance()->getOffset();
|
2018-07-13 15:56:37 +02:00
|
|
|
|
2022-02-02 10:29:30 +01:00
|
|
|
*time = offset + uptime;
|
2018-07-13 15:56:37 +02:00
|
|
|
|
2022-02-02 10:29:30 +01:00
|
|
|
return HasReturnvaluesIF::RETURN_OK;
|
2018-07-13 15:56:37 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
ReturnValue_t Clock::getUptime(timeval* uptime) {
|
2022-02-02 10:29:30 +01:00
|
|
|
*uptime = getUptime();
|
2018-07-13 15:56:37 +02:00
|
|
|
|
2022-02-02 10:29:30 +01:00
|
|
|
return HasReturnvaluesIF::RETURN_OK;
|
2018-07-13 15:56:37 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
timeval Clock::getUptime() {
|
2022-02-02 10:29:30 +01:00
|
|
|
TickType_t ticksSinceStart = xTaskGetTickCount();
|
|
|
|
return Timekeeper::ticksToTimeval(ticksSinceStart);
|
2018-07-13 15:56:37 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
ReturnValue_t Clock::getUptime(uint32_t* uptimeMs) {
|
2022-02-02 10:29:30 +01:00
|
|
|
timeval uptime = getUptime();
|
|
|
|
*uptimeMs = uptime.tv_sec * 1000 + uptime.tv_usec / 1000;
|
|
|
|
return HasReturnvaluesIF::RETURN_OK;
|
2018-07-13 15:56:37 +02:00
|
|
|
}
|
|
|
|
|
2022-02-02 10:29:30 +01:00
|
|
|
// uint32_t Clock::getUptimeSeconds() {
|
2020-12-14 11:17:22 +01:00
|
|
|
// timeval uptime = getUptime();
|
|
|
|
// return uptime.tv_sec;
|
2022-02-02 10:29:30 +01:00
|
|
|
// }
|
2020-12-14 11:17:22 +01:00
|
|
|
|
2018-07-13 15:56:37 +02:00
|
|
|
ReturnValue_t Clock::getClock_usecs(uint64_t* time) {
|
2022-02-02 10:29:30 +01:00
|
|
|
timeval time_timeval;
|
|
|
|
ReturnValue_t result = getClock_timeval(&time_timeval);
|
|
|
|
if (result != HasReturnvaluesIF::RETURN_OK) {
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
*time = time_timeval.tv_sec * 1000000 + time_timeval.tv_usec;
|
|
|
|
return HasReturnvaluesIF::RETURN_OK;
|
2018-07-13 15:56:37 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
ReturnValue_t Clock::getDateAndTime(TimeOfDay_t* time) {
|
2022-02-02 10:29:30 +01:00
|
|
|
timeval time_timeval;
|
|
|
|
ReturnValue_t result = getClock_timeval(&time_timeval);
|
|
|
|
if (result != HasReturnvaluesIF::RETURN_OK) {
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
struct tm time_tm;
|
2018-07-13 15:56:37 +02:00
|
|
|
|
2022-02-02 10:29:30 +01:00
|
|
|
gmtime_r(&time_timeval.tv_sec, &time_tm);
|
2018-07-13 15:56:37 +02:00
|
|
|
|
2022-02-02 10:29:30 +01:00
|
|
|
time->year = time_tm.tm_year + 1900;
|
|
|
|
time->month = time_tm.tm_mon + 1;
|
|
|
|
time->day = time_tm.tm_mday;
|
2018-07-13 15:56:37 +02:00
|
|
|
|
2022-02-02 10:29:30 +01:00
|
|
|
time->hour = time_tm.tm_hour;
|
|
|
|
time->minute = time_tm.tm_min;
|
|
|
|
time->second = time_tm.tm_sec;
|
2018-07-13 15:56:37 +02:00
|
|
|
|
2022-02-02 10:29:30 +01:00
|
|
|
time->usecond = time_timeval.tv_usec;
|
2018-07-13 15:56:37 +02:00
|
|
|
|
2022-02-02 10:29:30 +01:00
|
|
|
return HasReturnvaluesIF::RETURN_OK;
|
2018-07-13 15:56:37 +02:00
|
|
|
}
|
|
|
|
|
2022-02-02 10:29:30 +01:00
|
|
|
ReturnValue_t Clock::convertTimeOfDayToTimeval(const TimeOfDay_t* from, timeval* to) {
|
|
|
|
struct tm time_tm = {};
|
2018-07-13 15:56:37 +02:00
|
|
|
|
2022-02-02 10:29:30 +01:00
|
|
|
time_tm.tm_year = from->year - 1900;
|
|
|
|
time_tm.tm_mon = from->month - 1;
|
|
|
|
time_tm.tm_mday = from->day;
|
2018-07-13 15:56:37 +02:00
|
|
|
|
2022-02-02 10:29:30 +01:00
|
|
|
time_tm.tm_hour = from->hour;
|
|
|
|
time_tm.tm_min = from->minute;
|
|
|
|
time_tm.tm_sec = from->second;
|
2018-07-13 15:56:37 +02:00
|
|
|
|
2022-02-02 10:29:30 +01:00
|
|
|
time_t seconds = mktime(&time_tm);
|
2018-07-13 15:56:37 +02:00
|
|
|
|
2022-02-02 10:29:30 +01:00
|
|
|
to->tv_sec = seconds;
|
|
|
|
to->tv_usec = from->usecond;
|
|
|
|
// Fails in 2038..
|
|
|
|
return HasReturnvaluesIF::RETURN_OK;
|
2018-07-13 15:56:37 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
ReturnValue_t Clock::convertTimevalToJD2000(timeval time, double* JD2000) {
|
2022-02-02 10:29:30 +01:00
|
|
|
*JD2000 = (time.tv_sec - 946728000. + time.tv_usec / 1000000.) / 24. / 3600.;
|
|
|
|
return HasReturnvaluesIF::RETURN_OK;
|
2018-07-13 15:56:37 +02:00
|
|
|
}
|