4491d606df
-- 70f43a482d7d4ae4a255f17ca02b0106653dd600 by Shaindel Schwartz <shaindel@google.com>: Internal change PiperOrigin-RevId: 201571193 -- 93e6e9c2e683158be49d9dd1f5cb1a91d0c0f556 by Abseil Team <absl-team@google.com>: Internal change. PiperOrigin-RevId: 201567108 -- fbd8ee94fbe9f2448e5adf5e88706f9c8216048f by Juemin Yang <jueminyang@google.com>: str_format release PiperOrigin-RevId: 201565129 -- 387faa301555a8a888c4429df52734aa806dca46 by Abseil Team <absl-team@google.com>: Adds a defaulted allocator parameter to the size_type constructor of InlinedVector PiperOrigin-RevId: 201558711 -- 39b15ea2c68d7129d70cbde7e71af900032595ec by Matt Calabrese <calabrese@google.com>: Update the variant implementation to eliminate unnecessary checking on alternative access when the index is known or required to be correct. PiperOrigin-RevId: 201529535 -- adab77f1f7bb363aa534297f22aae2b0f08889ea by Abseil Team <absl-team@google.com>: Import of CCTZ from GitHub. PiperOrigin-RevId: 201458388 -- a701dc0ba62e3cadf0de14203415b91df4ee8151 by Greg Falcon <gfalcon@google.com>: Internal cleanup PiperOrigin-RevId: 201394836 -- 8a7191410b8f440fdfa27f722ff05e451502ab61 by Abseil Team <absl-team@google.com>: Import of CCTZ from GitHub. PiperOrigin-RevId: 201369269 GitOrigin-RevId: 70f43a482d7d4ae4a255f17ca02b0106653dd600 Change-Id: I8ab073b30b4e27405a3b6da2c826bb4f3f0b9af6
145 lines
4.1 KiB
C++
145 lines
4.1 KiB
C++
// Copyright 2016 Google Inc. All Rights Reserved.
|
|
//
|
|
// 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.
|
|
|
|
#include "absl/time/internal/cctz/include/cctz/time_zone.h"
|
|
|
|
#if defined(__ANDROID__)
|
|
#include <sys/system_properties.h>
|
|
#if __ANDROID_API__ >= 21
|
|
#include <dlfcn.h>
|
|
#endif
|
|
#endif
|
|
#include <cstdlib>
|
|
#include <cstring>
|
|
#include <string>
|
|
|
|
#include "time_zone_fixed.h"
|
|
#include "time_zone_impl.h"
|
|
|
|
namespace absl {
|
|
namespace time_internal {
|
|
namespace cctz {
|
|
|
|
#if defined(__ANDROID__) && __ANDROID_API__ >= 21
|
|
namespace {
|
|
// Android 'L' removes __system_property_get() from the NDK, however
|
|
// it is still a hidden symbol in libc so we use dlsym() to access it.
|
|
// See Chromium's base/sys_info_android.cc for a similar example.
|
|
|
|
using property_get_func = int (*)(const char*, char*);
|
|
|
|
property_get_func LoadSystemPropertyGet() {
|
|
int flag = RTLD_LAZY | RTLD_GLOBAL;
|
|
#if defined(RTLD_NOLOAD)
|
|
flag |= RTLD_NOLOAD; // libc.so should already be resident
|
|
#endif
|
|
if (void* handle = dlopen("libc.so", flag)) {
|
|
void* sym = dlsym(handle, "__system_property_get");
|
|
dlclose(handle);
|
|
return reinterpret_cast<property_get_func>(sym);
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
int __system_property_get(const char* name, char* value) {
|
|
static property_get_func system_property_get = LoadSystemPropertyGet();
|
|
return system_property_get ? system_property_get(name, value) : -1;
|
|
}
|
|
|
|
} // namespace
|
|
#endif
|
|
|
|
std::string time_zone::name() const {
|
|
return time_zone::Impl::get(*this).name();
|
|
}
|
|
|
|
time_zone::absolute_lookup time_zone::lookup(
|
|
const time_point<seconds>& tp) const {
|
|
return time_zone::Impl::get(*this).BreakTime(tp);
|
|
}
|
|
|
|
time_zone::civil_lookup time_zone::lookup(const civil_second& cs) const {
|
|
return time_zone::Impl::get(*this).MakeTime(cs);
|
|
}
|
|
|
|
bool operator==(time_zone lhs, time_zone rhs) {
|
|
return &time_zone::Impl::get(lhs) == &time_zone::Impl::get(rhs);
|
|
}
|
|
|
|
bool load_time_zone(const std::string& name, time_zone* tz) {
|
|
return time_zone::Impl::LoadTimeZone(name, tz);
|
|
}
|
|
|
|
time_zone utc_time_zone() {
|
|
return time_zone::Impl::UTC(); // avoid name lookup
|
|
}
|
|
|
|
time_zone fixed_time_zone(const seconds& offset) {
|
|
time_zone tz;
|
|
load_time_zone(FixedOffsetToName(offset), &tz);
|
|
return tz;
|
|
}
|
|
|
|
time_zone local_time_zone() {
|
|
const char* zone = ":localtime";
|
|
|
|
// Allow ${TZ} to override to default zone.
|
|
char* tz_env = nullptr;
|
|
#if defined(_MSC_VER)
|
|
_dupenv_s(&tz_env, nullptr, "TZ");
|
|
#else
|
|
tz_env = std::getenv("TZ");
|
|
#endif
|
|
#if defined(__ANDROID__)
|
|
char sysprop[PROP_VALUE_MAX];
|
|
if (tz_env == nullptr)
|
|
if (__system_property_get("persist.sys.timezone", sysprop) > 0)
|
|
tz_env = sysprop;
|
|
#endif
|
|
if (tz_env) zone = tz_env;
|
|
|
|
// We only support the "[:]<zone-name>" form.
|
|
if (*zone == ':') ++zone;
|
|
|
|
// Map "localtime" to a system-specific name, but
|
|
// allow ${LOCALTIME} to override the default name.
|
|
char* localtime_env = nullptr;
|
|
if (strcmp(zone, "localtime") == 0) {
|
|
#if defined(_MSC_VER)
|
|
// System-specific default is just "localtime".
|
|
_dupenv_s(&localtime_env, nullptr, "LOCALTIME");
|
|
#else
|
|
zone = "/etc/localtime"; // System-specific default.
|
|
localtime_env = std::getenv("LOCALTIME");
|
|
#endif
|
|
if (localtime_env) zone = localtime_env;
|
|
}
|
|
|
|
const std::string name = zone;
|
|
#if defined(_MSC_VER)
|
|
free(localtime_env);
|
|
free(tz_env);
|
|
#endif
|
|
|
|
time_zone tz;
|
|
load_time_zone(name, &tz); // Falls back to UTC.
|
|
// TODO: Follow the RFC3339 "Unknown Local Offset Convention" and
|
|
// arrange for %z to generate "-0000" when we don't know the local
|
|
// offset because the load_time_zone() failed and we're using UTC.
|
|
return tz;
|
|
}
|
|
|
|
} // namespace cctz
|
|
} // namespace time_internal
|
|
} // namespace absl
|