| /* |
| * |
| * Copyright (c) 2020 Project CHIP Authors |
| * Copyright (c) 2018 Nest Labs, Inc. |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| /** |
| * @file |
| * Provides default implementations for the platform Get/SetClock_ functions |
| * for POSIX and LwIP platforms. |
| */ |
| |
| // __STDC_LIMIT_MACROS must be defined for UINT8_MAX to be defined for pre-C++11 clib |
| #ifndef __STDC_LIMIT_MACROS |
| #define __STDC_LIMIT_MACROS |
| #endif // __STDC_LIMIT_MACROS |
| |
| // __STDC_CONSTANT_MACROS must be defined for INT64_C and UINT64_C to be defined for pre-C++11 clib |
| #ifndef __STDC_CONSTANT_MACROS |
| #define __STDC_CONSTANT_MACROS |
| #endif // __STDC_CONSTANT_MACROS |
| |
| // config |
| #include <system/SystemConfig.h> |
| |
| #if !CHIP_SYSTEM_CONFIG_PLATFORM_PROVIDES_TIME |
| |
| // module header |
| #include <system/SystemClock.h> |
| // common private |
| #include "SystemLayerPrivate.h" |
| |
| #include <support/CodeUtils.h> |
| #include <system/SystemError.h> |
| |
| #if CHIP_SYSTEM_CONFIG_USE_POSIX_TIME_FUNCTS |
| #include <time.h> |
| #if !(HAVE_CLOCK_GETTIME) |
| #include <sys/time.h> |
| #endif |
| #include <errno.h> |
| #endif // CHIP_SYSTEM_CONFIG_USE_POSIX_TIME_FUNCTS |
| |
| #if CHIP_SYSTEM_CONFIG_USE_LWIP |
| #include <lwip/sys.h> |
| #endif // CHIP_SYSTEM_CONFIG_USE_LWIP |
| |
| #include <stdint.h> |
| #include <stdlib.h> |
| |
| namespace chip { |
| namespace System { |
| namespace Platform { |
| namespace Layer { |
| |
| // -------------------- Default Get/SetClock Functions for POSIX Systems -------------------- |
| |
| #if CHIP_SYSTEM_CONFIG_USE_POSIX_TIME_FUNCTS |
| |
| #if !HAVE_CLOCK_GETTIME && !HAVE_GETTIMEOFDAY |
| #error "CHIP_SYSTEM_CONFIG_USE_POSIX_TIME_FUNCTS requires either clock_gettime() or gettimeofday()" |
| #endif |
| |
| #if HAVE_CLOCK_GETTIME |
| #if HAVE_DECL_CLOCK_BOOTTIME |
| // CLOCK_BOOTTIME is a Linux-specific option to clock_gettime for a clock which compensates for system sleep. |
| #define MONOTONIC_CLOCK_ID CLOCK_BOOTTIME |
| #define MONOTONIC_RAW_CLOCK_ID CLOCK_MONOTONIC_RAW |
| #else // HAVE_DECL_CLOCK_BOOTTIME |
| // CLOCK_MONOTONIC is defined in POSIX and hence is the default choice |
| #define MONOTONIC_CLOCK_ID CLOCK_MONOTONIC |
| #endif |
| #endif // HAVE_CLOCK_GETTIME |
| |
| uint64_t GetClock_Monotonic() |
| { |
| #if HAVE_CLOCK_GETTIME |
| struct timespec ts; |
| int res = clock_gettime(MONOTONIC_CLOCK_ID, &ts); |
| VerifyOrDie(res == 0); |
| return (static_cast<uint64_t>(ts.tv_sec) * UINT64_C(1000000)) + (static_cast<uint64_t>(ts.tv_nsec) / 1000); |
| #else // HAVE_CLOCK_GETTIME |
| struct timeval tv; |
| int res = gettimeofday(&tv, NULL); |
| VerifyOrDie(res == 0); |
| return (tv.tv_sec * UINT64_C(1000000)) + tv.tv_usec; |
| #endif // HAVE_CLOCK_GETTIME |
| } |
| |
| uint64_t GetClock_MonotonicMS() |
| { |
| return GetClock_Monotonic() / 1000; |
| } |
| |
| uint64_t GetClock_MonotonicHiRes() |
| { |
| #if HAVE_CLOCK_GETTIME && defined(MONOTONIC_RAW_CLOCK_ID) |
| struct timespec ts; |
| int res = clock_gettime(MONOTONIC_RAW_CLOCK_ID, &ts); |
| VerifyOrDie(res == 0); |
| return (ts.tv_sec * UINT64_C(1000000)) + (ts.tv_nsec / 1000); |
| #else // HAVE_CLOCK_GETTIME |
| return GetClock_Monotonic(); |
| #endif // HAVE_CLOCK_GETTIME |
| } |
| |
| Error GetClock_RealTime(uint64_t & curTime) |
| { |
| #if HAVE_CLOCK_GETTIME |
| struct timespec ts; |
| int res = clock_gettime(CLOCK_REALTIME, &ts); |
| if (res != 0) |
| { |
| return MapErrorPOSIX(errno); |
| } |
| if (ts.tv_sec < CHIP_SYSTEM_CONFIG_VALID_REAL_TIME_THRESHOLD) |
| { |
| return CHIP_SYSTEM_ERROR_REAL_TIME_NOT_SYNCED; |
| } |
| curTime = (static_cast<uint64_t>(ts.tv_sec) * UINT64_C(1000000)) + (static_cast<uint64_t>(ts.tv_nsec) / 1000); |
| return CHIP_SYSTEM_NO_ERROR; |
| #else // HAVE_CLOCK_GETTIME |
| struct timeval tv; |
| int res = gettimeofday(&tv, NULL); |
| if (res != 0) |
| { |
| return MapErrorPOSIX(errno); |
| } |
| if (tv.tv_sec < CHIP_SYSTEM_CONFIG_VALID_REAL_TIME_THRESHOLD) |
| { |
| return CHIP_SYSTEM_ERROR_REAL_TIME_NOT_SYNCED; |
| } |
| curTime = (tv.tv_sec * UINT64_C(1000000)) + tv.tv_usec; |
| return CHIP_SYSTEM_NO_ERROR; |
| #endif // HAVE_CLOCK_GETTIME |
| } |
| |
| Error GetClock_RealTimeMS(uint64_t & curTime) |
| { |
| Error err = GetClock_RealTime(curTime); |
| curTime = curTime / 1000; |
| return err; |
| } |
| |
| #if HAVE_CLOCK_SETTIME || HAVE_SETTIMEOFDAY |
| |
| Error SetClock_RealTime(uint64_t newCurTime) |
| { |
| #if HAVE_CLOCK_SETTIME |
| struct timespec ts; |
| ts.tv_sec = static_cast<time_t>(newCurTime / UINT64_C(1000000)); |
| ts.tv_nsec = static_cast<long>(newCurTime % UINT64_C(1000000)) * 1000; |
| int res = clock_settime(CLOCK_REALTIME, &ts); |
| if (res != 0) |
| { |
| return (errno == EPERM) ? CHIP_SYSTEM_ERROR_ACCESS_DENIED : MapErrorPOSIX(errno); |
| } |
| return CHIP_SYSTEM_NO_ERROR; |
| #else // HAVE_CLOCK_SETTIME |
| struct timeval tv; |
| tv.tv_sec = static_cast<time_t>(newCurTime / UINT64_C(1000000)); |
| tv.tv_usec = static_cast<long>(newCurTime % UINT64_C(1000000)); |
| int res = settimeofday(&tv, NULL); |
| if (res != 0) |
| { |
| return (errno == EPERM) ? CHIP_SYSTEM_ERROR_ACCESS_DENIED : MapErrorPOSIX(errno); |
| } |
| return CHIP_SYSTEM_NO_ERROR; |
| #endif // HAVE_CLOCK_SETTIME |
| } |
| |
| #else // !HAVE_CLOCK_SETTTIME |
| |
| Error SetClock_RealTime(uint64_t newCurTime) |
| { |
| return CHIP_SYSTEM_ERROR_NOT_SUPPORTED; |
| } |
| |
| #endif // HAVE_CLOCK_SETTIME || HAVE_SETTIMEOFDAY |
| |
| #endif // CHIP_SYSTEM_CONFIG_USE_POSIX_TIME_FUNCTS |
| |
| // -------------------- Default Get/SetClock Functions for LwIP Systems -------------------- |
| |
| #if CHIP_SYSTEM_CONFIG_USE_LWIP_MONOTONIC_TIME |
| |
| uint64_t GetClock_Monotonic(void) |
| { |
| return GetClock_MonotonicMS(); |
| } |
| |
| uint64_t GetClock_MonotonicMS(void) |
| { |
| static volatile uint64_t overflow = 0; |
| static volatile u32_t lastSample = 0; |
| static volatile uint8_t lock = 0; |
| static const uint64_t kOverflowIncrement = static_cast<uint64_t>(0x100000000); |
| |
| uint64_t overflowSample; |
| u32_t sample; |
| |
| // Tracking timer wrap assumes that this function gets called with |
| // a period that is less than 1/2 the timer range. |
| if (__sync_bool_compare_and_swap(&lock, 0, 1)) |
| { |
| sample = sys_now(); |
| |
| if (lastSample > sample) |
| { |
| overflow += kOverflowIncrement; |
| } |
| |
| lastSample = sample; |
| overflowSample = overflow; |
| |
| __sync_bool_compare_and_swap(&lock, 1, 0); |
| } |
| else |
| { |
| // a lower priority task is in the block above. Depending where that |
| // lower task is blocked can spell trouble in a timer wrap condition. |
| // the question here is what this task should use as an overflow value. |
| // To fix this race requires a platform api that can be used to |
| // protect critical sections. |
| overflowSample = overflow; |
| sample = sys_now(); |
| } |
| |
| return static_cast<uint64_t>(overflowSample | static_cast<uint64_t>(sample)); |
| } |
| |
| uint64_t GetClock_MonotonicHiRes(void) |
| { |
| return GetClock_MonotonicMS(); |
| } |
| |
| Error GetClock_RealTime(uint64_t & curTime) |
| { |
| return CHIP_SYSTEM_ERROR_NOT_SUPPORTED; |
| } |
| |
| Error GetClock_RealTimeMS(uint64_t & curTime) |
| { |
| return CHIP_SYSTEM_ERROR_NOT_SUPPORTED; |
| } |
| |
| Error SetClock_RealTime(uint64_t newCurTime) |
| { |
| return CHIP_SYSTEM_ERROR_NOT_SUPPORTED; |
| } |
| |
| #endif // CHIP_SYSTEM_CONFIG_USE_LWIP_MONOTONIC_TIME |
| |
| } // namespace Layer |
| } // namespace Platform |
| } // namespace System |
| } // namespace chip |
| |
| #endif // CHIP_SYSTEM_CONFIG_PLATFORM_PROVIDES_TIME |