| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163 |
- /*-
- * Copyright © 2025 ARJANEN Loïc Jean David
- * All rights reserved.
- *
- * Redistribution and use in source and binary forms, with or without
- * modification, are permitted provided that the following conditions
- * are met:
- * 1. Redistributions of source code must retain the above copyright
- * notice, this list of conditions and the following disclaimer.
- * 2. Redistributions in binary form must reproduce the above copyright
- * notice, this list of conditions and the following disclaimer in the
- * documentation and/or other materials provided with the distribution.
- *
- * THIS SOFTWARE IS PROVIDED BY THE AUTHOR(S) ``AS IS'' AND ANY EXPRESS OR
- * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
- * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
- * IN NO EVENT SHALL THE AUTHOR(S) BE LIABLE FOR ANY DIRECT, INDIRECT,
- * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
- * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
- * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
- * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
- * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
- * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
- */
- #include "archive_platform.h"
- #include "archive_private.h"
- #include "archive_time_private.h"
- #include <limits.h>
- #include <stdlib.h>
- #include <string.h>
- #define NTFS_EPOC_TIME ARCHIVE_LITERAL_ULL(11644473600)
- #define NTFS_TICKS ARCHIVE_LITERAL_ULL(10000000)
- #define NTFS_EPOC_TICKS (NTFS_EPOC_TIME * NTFS_TICKS)
- #define DOS_MIN_TIME 0x00210000U
- #define DOS_MAX_TIME 0xff9fbf7dU
- #if defined(_WIN32) && !defined(__CYGWIN__)
- #include <winnt.h>
- /* Windows FILETIME to NTFS time. */
- uint64_t
- FILETIME_to_ntfs(const FILETIME* filetime)
- {
- ULARGE_INTEGER utc;
- utc.HighPart = filetime->dwHighDateTime;
- utc.LowPart = filetime->dwLowDateTime;
- return utc.QuadPart;
- }
- #endif
- /* Convert an MSDOS-style date/time into Unix-style time. */
- int64_t
- dos_to_unix(uint32_t dos_time)
- {
- uint16_t msTime, msDate;
- struct tm ts;
- time_t t;
- msTime = (0xFFFF & dos_time);
- msDate = (dos_time >> 16);
- memset(&ts, 0, sizeof(ts));
- ts.tm_year = ((msDate >> 9) & 0x7f) + 80; /* Years since 1900. */
- ts.tm_mon = ((msDate >> 5) & 0x0f) - 1; /* Month number. */
- ts.tm_mday = msDate & 0x1f; /* Day of month. */
- ts.tm_hour = (msTime >> 11) & 0x1f;
- ts.tm_min = (msTime >> 5) & 0x3f;
- ts.tm_sec = (msTime << 1) & 0x3e;
- ts.tm_isdst = -1;
- t = mktime(&ts);
- return (int64_t)(t == (time_t)-1 ? INT32_MAX : t);
- }
- /* Convert into MSDOS-style date/time. */
- uint32_t
- unix_to_dos(int64_t unix_time)
- {
- struct tm *t;
- uint32_t dt;
- time_t ut = unix_time;
- #if defined(HAVE_LOCALTIME_R) || defined(HAVE_LOCALTIME_S)
- struct tm tmbuf;
- #endif
- if (sizeof(time_t) < sizeof(int64_t) && (int64_t)ut != unix_time) {
- ut = (time_t)(unix_time > 0 ? INT32_MAX : INT32_MIN);
- }
- #if defined(HAVE_LOCALTIME_S)
- t = localtime_s(&tmbuf, &ut) ? NULL : &tmbuf;
- #elif defined(HAVE_LOCALTIME_R)
- t = localtime_r(&ut, &tmbuf);
- #else
- t = localtime(&ut);
- #endif
- dt = 0;
- if (t != NULL && t->tm_year >= INT_MIN + 80) {
- const int year = t->tm_year - 80;
- if (year & ~0x7f) {
- dt = year > 0 ? DOS_MAX_TIME : DOS_MIN_TIME;
- }
- else {
- dt += (year & 0x7f) << 9;
- dt += ((t->tm_mon + 1) & 0x0f) << 5;
- dt += (t->tm_mday & 0x1f);
- dt <<= 16;
- dt += (t->tm_hour & 0x1f) << 11;
- dt += (t->tm_min & 0x3f) << 5;
- /* Only counting every 2 seconds. */
- dt += (t->tm_sec & 0x3e) >> 1;
- }
- }
- if (dt > DOS_MAX_TIME) {
- dt = DOS_MAX_TIME;
- }
- else if (dt < DOS_MIN_TIME) {
- dt = DOS_MIN_TIME;
- }
- return dt;
- }
- /* Convert NTFS time to Unix sec/nsec */
- void
- ntfs_to_unix(uint64_t ntfs, int64_t* secs, uint32_t* nsecs)
- {
- if (ntfs > INT64_MAX) {
- ntfs -= NTFS_EPOC_TICKS;
- *secs = ntfs / NTFS_TICKS;
- *nsecs = 100 * (ntfs % NTFS_TICKS);
- }
- else {
- lldiv_t tdiv;
- int64_t value = (int64_t)ntfs - (int64_t)NTFS_EPOC_TICKS;
- tdiv = lldiv(value, NTFS_TICKS);
- *secs = tdiv.quot;
- *nsecs = (uint32_t)(tdiv.rem * 100);
- }
- }
- /* Convert Unix sec/nsec to NTFS time */
- uint64_t
- unix_to_ntfs(int64_t secs, uint32_t nsecs)
- {
- uint64_t ntfs;
- if (secs < -(int64_t)NTFS_EPOC_TIME)
- return 0;
- ntfs = secs + NTFS_EPOC_TIME;
- if (ntfs > UINT64_MAX / NTFS_TICKS)
- return UINT64_MAX;
- ntfs *= NTFS_TICKS;
- if (ntfs > UINT64_MAX - nsecs/100)
- return UINT64_MAX;
- return ntfs + nsecs/100;
- }
|