mirror of
https://github.com/HowardHinnant/date.git
synced 2025-01-14 01:37:57 +08:00
Do a runtime test for realpath vs readlink
* Ubuntu needs to use readlink under current_zone and most everyone else needs realpath. Attempting to make everyone happy with this commit.
This commit is contained in:
parent
9502bc27a3
commit
5f49afc1e9
83
src/tz.cpp
83
src/tz.cpp
@ -3673,6 +3673,56 @@ tzdb::current_zone() const
|
|||||||
|
|
||||||
#else // !_WIN32
|
#else // !_WIN32
|
||||||
|
|
||||||
|
#if HAS_STRING_VIEW
|
||||||
|
|
||||||
|
static
|
||||||
|
std::string_view
|
||||||
|
extract_tz_name(char const* rp)
|
||||||
|
{
|
||||||
|
using namespace std;
|
||||||
|
string_view result = rp;
|
||||||
|
CONSTDATA string_view zoneinfo = "zoneinfo";
|
||||||
|
size_t pos = result.rfind(zoneinfo);
|
||||||
|
if (pos == result.npos)
|
||||||
|
throw runtime_error(
|
||||||
|
"current_zone() failed to find \"zoneinfo\" in " + string(result));
|
||||||
|
pos = result.find('/', pos);
|
||||||
|
result.remove_prefix(pos + 1);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
#else // !HAS_STRING_VIEW
|
||||||
|
|
||||||
|
static
|
||||||
|
std::string
|
||||||
|
extract_tz_name(char const* rp)
|
||||||
|
{
|
||||||
|
using namespace std;
|
||||||
|
string result = rp;
|
||||||
|
CONSTDATA char zoneinfo[] = "zoneinfo";
|
||||||
|
size_t pos = result.rfind(zoneinfo);
|
||||||
|
if (pos == result.npos)
|
||||||
|
throw runtime_error(
|
||||||
|
"current_zone() failed to find \"zoneinfo\" in " + result);
|
||||||
|
pos = result.find('/', pos);
|
||||||
|
result.erase(0, pos + 1);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif // HAS_STRING_VIEW
|
||||||
|
|
||||||
|
static
|
||||||
|
bool
|
||||||
|
sniff_realpath(const char* timezone)
|
||||||
|
{
|
||||||
|
using namespace std;
|
||||||
|
char rp[PATH_MAX+1] = {};
|
||||||
|
if (realpath(timezone, rp) == nullptr)
|
||||||
|
throw system_error(errno, system_category(), "realpath() failed");
|
||||||
|
auto result = extract_tz_name(rp);
|
||||||
|
return result != "posixrules";
|
||||||
|
}
|
||||||
|
|
||||||
const time_zone*
|
const time_zone*
|
||||||
tzdb::current_zone() const
|
tzdb::current_zone() const
|
||||||
{
|
{
|
||||||
@ -3692,31 +3742,22 @@ tzdb::current_zone() const
|
|||||||
{
|
{
|
||||||
struct stat sb;
|
struct stat sb;
|
||||||
CONSTDATA auto timezone = "/etc/localtime";
|
CONSTDATA auto timezone = "/etc/localtime";
|
||||||
if (lstat(timezone, &sb) == 0 && S_ISLNK(sb.st_mode) && sb.st_size > 0) {
|
if (lstat(timezone, &sb) == 0 && S_ISLNK(sb.st_mode) && sb.st_size > 0)
|
||||||
|
{
|
||||||
using namespace std;
|
using namespace std;
|
||||||
|
static const bool use_realpath = sniff_realpath(timezone);
|
||||||
char rp[PATH_MAX+1] = {};
|
char rp[PATH_MAX+1] = {};
|
||||||
|
if (use_realpath)
|
||||||
|
{
|
||||||
if (realpath(timezone, rp) == nullptr)
|
if (realpath(timezone, rp) == nullptr)
|
||||||
throw system_error(errno, system_category(), "realpath() failed");
|
throw system_error(errno, system_category(), "realpath() failed");
|
||||||
#if HAS_STRING_VIEW
|
}
|
||||||
string_view result = rp;
|
else
|
||||||
CONSTDATA string_view zoneinfo = "zoneinfo";
|
{
|
||||||
size_t pos = result.rfind(zoneinfo);
|
if (readlink(timezone, rp, sizeof(rp)-1) <= 0)
|
||||||
if (pos == result.npos)
|
throw system_error(errno, system_category(), "readlink() failed");
|
||||||
throw runtime_error(
|
}
|
||||||
"current_zone() failed to find \"zoneinfo\" in " + string(result));
|
return locate_zone(extract_tz_name(rp));
|
||||||
pos = result.find('/', pos);
|
|
||||||
result.remove_prefix(pos + 1);
|
|
||||||
#else
|
|
||||||
string result = rp;
|
|
||||||
CONSTDATA char zoneinfo[] = "zoneinfo";
|
|
||||||
size_t pos = result.rfind(zoneinfo);
|
|
||||||
if (pos == result.npos)
|
|
||||||
throw runtime_error(
|
|
||||||
"current_zone() failed to find \"zoneinfo\" in " + result);
|
|
||||||
pos = result.find('/', pos);
|
|
||||||
result.erase(0, pos + 1);
|
|
||||||
#endif
|
|
||||||
return locate_zone(result);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// On embedded systems e.g. buildroot with uclibc the timezone is linked
|
// On embedded systems e.g. buildroot with uclibc the timezone is linked
|
||||||
|
Loading…
x
Reference in New Issue
Block a user