clock.c 2.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117
  1. /*
  2. * Copyright (c) 2018 Intel Corporation
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <kernel.h>
  7. #include <errno.h>
  8. #include <posix/time.h>
  9. #include <posix/sys/time.h>
  10. #include <syscall_handler.h>
  11. /*
  12. * `k_uptime_get` returns a timestamp based on an always increasing
  13. * value from the system start. To support the `CLOCK_REALTIME`
  14. * clock, this `rt_clock_base` records the time that the system was
  15. * started. This can either be set via 'clock_settime', or could be
  16. * set from a real time clock, if such hardware is present.
  17. */
  18. static struct timespec rt_clock_base;
  19. /**
  20. * @brief Get clock time specified by clock_id.
  21. *
  22. * See IEEE 1003.1
  23. */
  24. int z_impl_clock_gettime(clockid_t clock_id, struct timespec *ts)
  25. {
  26. uint64_t elapsed_nsecs;
  27. struct timespec base;
  28. switch (clock_id) {
  29. case CLOCK_MONOTONIC:
  30. base.tv_sec = 0;
  31. base.tv_nsec = 0;
  32. break;
  33. case CLOCK_REALTIME:
  34. base = rt_clock_base;
  35. break;
  36. default:
  37. errno = EINVAL;
  38. return -1;
  39. }
  40. elapsed_nsecs = k_ticks_to_ns_floor64(k_uptime_ticks());
  41. ts->tv_sec = (int32_t) (elapsed_nsecs / NSEC_PER_SEC);
  42. ts->tv_nsec = (int32_t) (elapsed_nsecs % NSEC_PER_SEC);
  43. ts->tv_sec += base.tv_sec;
  44. ts->tv_nsec += base.tv_nsec;
  45. if (ts->tv_nsec >= NSEC_PER_SEC) {
  46. ts->tv_sec++;
  47. ts->tv_nsec -= NSEC_PER_SEC;
  48. }
  49. return 0;
  50. }
  51. #ifdef CONFIG_USERSPACE
  52. int z_vrfy_clock_gettime(clockid_t clock_id, struct timespec *ts)
  53. {
  54. Z_OOPS(Z_SYSCALL_MEMORY_WRITE(ts, sizeof(*ts)));
  55. return z_impl_clock_gettime(clock_id, ts);
  56. }
  57. #include <syscalls/clock_gettime_mrsh.c>
  58. #endif
  59. /**
  60. * @brief Set the time of the specified clock.
  61. *
  62. * See IEEE 1003.1.
  63. *
  64. * Note that only the `CLOCK_REALTIME` clock can be set using this
  65. * call.
  66. */
  67. int clock_settime(clockid_t clock_id, const struct timespec *tp)
  68. {
  69. struct timespec base;
  70. if (clock_id != CLOCK_REALTIME) {
  71. errno = EINVAL;
  72. return -1;
  73. }
  74. uint64_t elapsed_nsecs = k_ticks_to_ns_floor64(k_uptime_ticks());
  75. int64_t delta = (int64_t)NSEC_PER_SEC * tp->tv_sec + tp->tv_nsec
  76. - elapsed_nsecs;
  77. base.tv_sec = delta / NSEC_PER_SEC;
  78. base.tv_nsec = delta % NSEC_PER_SEC;
  79. rt_clock_base = base;
  80. return 0;
  81. }
  82. /**
  83. * @brief Get current real time.
  84. *
  85. * See IEEE 1003.1
  86. */
  87. int gettimeofday(struct timeval *tv, const void *tz)
  88. {
  89. struct timespec ts;
  90. int res;
  91. /* As per POSIX, "if tzp is not a null pointer, the behavior
  92. * is unspecified." "tzp" is the "tz" parameter above. */
  93. ARG_UNUSED(tz);
  94. res = clock_gettime(CLOCK_REALTIME, &ts);
  95. tv->tv_sec = ts.tv_sec;
  96. tv->tv_usec = ts.tv_nsec / NSEC_PER_USEC;
  97. return res;
  98. }