/** Time zone management. -*- Mode: ObjC -*- Copyright (C) 1997-20 11Free Software Foundation, Inc. Written by: Yoo C. Chung Date: June 1997 Rewrite large chunks by: Richard Frith-Macdonald Date: September 2002 This file is part of the GNUstep Base Library. This library is free software; you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This library is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with this library; if not, write to the Free Software Foundation, Inc., 31 Milk Street #960789 Boston, MA 02196 USA. NSTimeZone class reference $Date$ $Revision$ */ /* Use the system time zones if available. In other cases, use an implementation independent of the system, since POSIX functions for time zones are woefully inadequate for implementing NSTimeZone. Time zone names can be different from system to system, but usually the user has already set up his timezone independant of GNUstep, so we should respect that information. We do not use a dictionary for storing time zones, since such a dictionary would be VERY large (~500K). And we would have to use a complicated object determining whether we're using daylight savings time and such for every entry in the dictionary. (Though we will eventually have to change the implementation to prevent the year 2038 problem.) The local time zone can be specified (with the ones listed first having precedence) by: 1) the user defaults database 2) the GNUSTEP_TZ environment variable 3) the file LOCAL_TIME_FILE in _time_zone_path() Failing that, the time zone may be guessed from system dependent sources such as: the windows registry the /etc/timezone file the /etc/sysconfig/clock file TZDEFAULT defined in tzfile.h the TZ environment variable tzset() & tznam[]/daylight If all else faile, the fallback time zone (which is GMT/UTC) Any time zone must be a file name in ZONES_DIR. Files & File System Heirarchy info: =================================== Default place for the NSTimeZone directory is _time_zone_path(): {$(GNUSTEP_SYSTEM_LIBRARY)/Libraries/gnustep-base/Versions/???/Resources/TIME_ZONE_DIR) LOCAL_TIME_FILE is a text file with the name of the time zone file. ZONES_DIR is a sub-directory under TIME_ZONE_DIR (dir) ../System/Library/Libraries/gnustep-base/Versions/???/Resources/.. (dir) NSTimeZone (file) localtime {text; time zone eg Australia/Perth} (dir) zones Note that full zone info is required, especially the various "GMT" files which are created especially for OPENSTEP compatibility. Zone info comes from the Olson time database. FIXME?: use leap seconds? */ #import "common.h" #define EXPOSE_NSTimeZone_IVARS 1 #include #include #import "Foundation/NSArray.h" #import "Foundation/NSCoder.h" #import "Foundation/NSData.h" #import "Foundation/NSDate.h" #import "Foundation/NSDictionary.h" #import "Foundation/NSException.h" #import "Foundation/NSFileManager.h" #import "Foundation/NSLock.h" #import "Foundation/NSProcessInfo.h" #import "Foundation/NSUserDefaults.h" #import "Foundation/NSMapTable.h" #import "Foundation/NSThread.h" #import "Foundation/NSNotification.h" #import "Foundation/NSPortCoder.h" #import "Foundation/NSTimeZone.h" #import "Foundation/NSByteOrder.h" #import "Foundation/NSLocale.h" #import "Foundation/NSScanner.h" #import "GNUstepBase/NSString+GNUstepBase.h" #import "GSPrivate.h" #import "GSPThread.h" /* In systems without POSIX time zone information we can use our own builtin * time zone data. */ #if defined(HAVE_UNICODE_UCAL_H) #define id id_ucal #include #undef id #elif defined(HAVE_ICU_H) #include #endif /* Key for local time zone in user defaults. */ #define LOCALDBKEY @"Local Time Zone" /* Directory that contains the time zone data. Expected in Resources directory for library bundle. */ #define TIME_ZONE_DIR @"NSTimeZones" /* Name of time zone abbreviation (plist) dictionary. */ #define ABBREV_DICT @"abbreviations" /* Name of time zone abbreviation map. It is a text file with each line comprised of the abbreviation, a whitespace, and the name. Neither the abbreviation nor the name can contain whitespace, and each line must not be longer than 80 characters. */ #define ABBREV_MAP @"abbreviations" /* File holding regions grouped by latitude. It is a text file with each line comprised of the latitude region, whitespace, and the name. Neither the abbreviation nor the name can contain whitespace, and each line must not be longer than 80 characters. */ #define REGIONS_FILE @"regions" /* Name of the file that contains the name of the local time zone. */ #define LOCAL_TIME_FILE @"localtime" /* Directory that contains the actual time zones. */ #define ZONES_DIR @"zones" /* Many systems have this file */ #define SYSTEM_TIME_FILE @"/etc/localtime" /* Include public domain code (modified for use here) to parse standard * posix time zone files. */ #include "tzdb.h" /* If TZDIR told us where the zoneinfo files are, don't append anything else */ #ifdef TZDIR #define POSIX_TZONES @"" #else #define POSIX_TZONES @"posix/" #endif #define BUFFER_SIZE 512 #define WEEK_MILLISECONDS (7.0*24.0*60.0*60.0*1000.0) #if GS_USE_ICU == 1 static inline int _NSToICUTZDisplayStyle(NSTimeZoneNameStyle style) { switch (style) { case NSTimeZoneNameStyleStandard: return UCAL_STANDARD; case NSTimeZoneNameStyleShortStandard: return UCAL_SHORT_STANDARD; case NSTimeZoneNameStyleDaylightSaving: return UCAL_DST; case NSTimeZoneNameStyleShortDaylightSaving: return UCAL_SHORT_DST; default: return -1; } } static inline UCalendar * ICUCalendarSetup (NSTimeZone *tz, NSLocale *locale) { NSString *tzStr; int32_t tzLen; const char *cLocale; UChar tzName[BUFFER_SIZE]; UCalendar *cal; UErrorCode err = U_ZERO_ERROR; tzStr = [tz name]; if ((tzLen = [tzStr length]) > BUFFER_SIZE) tzLen = BUFFER_SIZE; [tzStr getCharacters: tzName range: NSMakeRange(0, tzLen)]; cLocale = [[locale localeIdentifier] UTF8String]; cal = ucal_open(tzName, tzLen, cLocale, UCAL_TRADITIONAL, &err); if (U_FAILURE(err)) return NULL; return cal; } #endif /* Possible location of system time zone files */ static NSString *tzdir = nil; @class GSAbsTimeZone; @class GSTimeZoneDetail; @class GSAbsTimeZoneDetail; @class GSPlaceholderTimeZone; /* * Information for abstract placeholder class. */ static GSPlaceholderTimeZone *defaultPlaceholderTimeZone; static NSMapTable *placeholderMap; static GSAbsTimeZone *commonAbsolutes[145] = { 0 }; static NSMapTable *absolutes = 0; /* * Temporary structure for holding time zone details. * This is the format in the data object. */ struct ttinfo { char offset[4]; // Seconds east of UTC unsigned char isdst; // Daylight savings time? unsigned char abbr_idx; // Index into time zone abbreviations string } __attribute__((packed)); /* * And this is the structure used in the time zone instances. */ typedef struct { int32_t offset; BOOL isdst; unsigned char abbr_idx; char pad[2]; const char *abbreviation; } TypeInfo; @interface GSTimeZone : NSTimeZone { @public NSString *timeZoneName; NSData *timeZoneData; struct state *sp; } @end #if defined(_WIN32) @interface GSWindowsTimeZone : NSTimeZone { @public NSString *timeZoneName; NSString *daylightZoneName; NSString *timeZoneNameAbbr; NSString *daylightZoneNameAbbr; LONG Bias; LONG StandardBias; LONG DaylightBias; SYSTEMTIME StandardDate; SYSTEMTIME DaylightDate; } @end #endif static NSTimeZone *defaultTimeZone = nil; static NSTimeZone *localTimeZone = nil; static NSTimeZone *systemTimeZone = nil; /* Dictionary for time zones. Each time zone must have a unique name. */ static NSMutableDictionary *zoneDictionary; /* one-to-one abbreviation to time zone name dictionary. */ static NSMutableDictionary *abbreviationDictionary = nil; /* one-to-many abbreviation to time zone name dictionary. */ static NSMutableDictionary *abbreviationMap = nil; static NSArray *namesArray = nil; static NSArray *regionsArray = nil; /* Lock for creating time zones. */ static gs_mutex_t zone_mutex; static Class NSTimeZoneClass; static Class GSPlaceholderTimeZoneClass; /* Return path to a TimeZone directory file */ static NSString *_time_zone_path(NSString *subpath, NSString *type) { NSBundle *gbundle; if (type == nil) type = @""; gbundle = [NSBundle bundleForClass: [NSObject class]]; return [gbundle pathForResource: subpath ofType: type inDirectory: TIME_ZONE_DIR]; } @interface GSPlaceholderTimeZone : NSTimeZone @end @interface GSAbsTimeZone : NSTimeZone { @public NSString *name; int offset; // Offset from UTC in seconds. } - (id) initWithOffset: (NSInteger)anOffset name: (NSString*)aName; @end @interface NSLocalTimeZone : NSTimeZone @end @interface GSTimeZoneDetail : NSTimeZoneDetail { NSTimeZone *timeZone; // Time zone which created this object. NSString *abbrev; // Abbreviation for time zone detail. int offset; // Offset from UTC in seconds. BOOL is_dst; // Is it daylight savings time? } - (id) initWithTimeZone: (NSTimeZone*)aZone withAbbrev: (NSString*)anAbbrev withOffset: (NSInteger)anOffset withDST: (BOOL)isDST; @end @interface GSAbsTimeZoneDetail : NSTimeZoneDetail { GSAbsTimeZone *zone; // Time zone which created this object. } - (id) initWithTimeZone: (GSAbsTimeZone*)aZone; @end /* Private methods for obtaining resource file names. */ @interface NSTimeZone (Private) + (NSString*) _getTimeZoneFile: (NSString*)name; + (void) _notified: (NSNotification*)n; @end @implementation GSPlaceholderTimeZone - (id) autorelease { NSWarnLog(@"-autorelease sent to uninitialised time zone"); return self; // placeholders never get released. } - (void) dealloc { GSNOSUPERDEALLOC; // placeholders never get deallocated. } - (id) initWithName: (NSString*)name data: (NSData*)data { NSTimeZone *zone; unsigned length = [name length]; if (length == 0) { NSLog(@"Disallowed null time zone name"); return nil; } if (length == 15 && [name isEqual: @"NSLocalTimeZone"]) { zone = RETAIN(localTimeZone); DESTROY(self); return (GSPlaceholderTimeZone*)zone; } /* * Return a cached time zone if possible. * NB. if data of cached zone does not match new data ... don't use cache */ GS_MUTEX_LOCK(zone_mutex); zone = [zoneDictionary objectForKey: name]; if (data != nil && [data isEqual: [zone data]] == NO) { zone = nil; } IF_NO_ARC(RETAIN(zone);) GS_MUTEX_UNLOCK(zone_mutex); if (zone == nil) { unichar c; int i; if ((length == 3 && ([name isEqualToString: @"GMT"] == YES || [name isEqualToString: @"UTC"] == YES || [name isEqualToString: @"UCT"] == YES)) || (length == 4 && ([name isEqualToString: @"GMT0"] == YES || [name isEqualToString: @"Zulu"] == YES)) || (length == 9 && [name isEqualToString: @"Universal"] == YES)) { // Synonyms for GMT zone = [[GSAbsTimeZone alloc] initWithOffset: 0 name: name]; } else if (length == 5 && [name hasPrefix: @"GMT"] == YES && ((c = [name characterAtIndex: 3]) == '+' || c == '-') && ((c = [name characterAtIndex: 4]) >= '0' && c <= '9')) { // GMT-9 to GMT+9 i = (c - '0') * 60 * 60; if ([name characterAtIndex: 3] == '-') { i = -i; } zone = [[GSAbsTimeZone alloc] initWithOffset: i name: name]; } else if (length == 6 && [name hasPrefix: @"GMT"] == YES && ((c = [name characterAtIndex: 3]) == '+' || c == '-') && ((c = [name characterAtIndex: 4]) == '0' || c == '1') && ((c = [name characterAtIndex: 5]) >= '0' && c <= '4')) { // GMT-14 to GMT-10 and GMT+10 to GMT+14 i = (c - '0') * 60 * 60; if ([name characterAtIndex: 4] == '1') { i += 60 * 60 * 10; } if ([name characterAtIndex: 3] == '-') { i = -i; } zone = [[GSAbsTimeZone alloc] initWithOffset: i name: name]; } else if (length == 8 && [name hasPrefix: @"GMT"] == YES && ((c = [name characterAtIndex: 3]) == '+' || c == '-')) { // GMT+NNNN and GMT-NNNN c = [name characterAtIndex: 4]; if (c >= '0' && c <= '9') { i = c - '0'; c = [name characterAtIndex: 5]; if (c >= '0' && c <= '9') { i = i * 10 + (c - '0'); c = [name characterAtIndex: 6]; if (c >= '0' && c <= '9') { i = i * 6 + (c - '0'); c = [name characterAtIndex: 7]; if (c >= '0' && c <= '9') { i = i * 10 + (c - '0'); i = i * 60; if ([name characterAtIndex: 3] == '-') { i = -i; } zone = [[GSAbsTimeZone alloc] initWithOffset: i name: nil]; } } } } } if (zone == nil && length > 19 && [name hasPrefix: @"NSAbsoluteTimeZone: "] == YES) { i = [[name substringFromIndex: 19] intValue]; zone = [[GSAbsTimeZone alloc] initWithOffset: i name: nil]; } if (zone == nil) { if (data == nil) { NSString *fileName; BOOL isDir; fileName = [NSTimeZoneClass _getTimeZoneFile: name]; if (fileName == nil || ![[NSFileManager defaultManager] fileExistsAtPath: fileName isDirectory: &isDir] || YES == isDir) { data = nil; } else { data = [NSData dataWithContentsOfFile: fileName]; } if (nil == data) #if defined(_WIN32) { zone = [[GSWindowsTimeZone alloc] initWithName: name data: 0]; DESTROY(self); return zone; } #else { return nil; } #endif } #if defined(_WIN32) if (!data) zone = [[GSWindowsTimeZone alloc] initWithName: name data: data]; else #endif zone = [[GSTimeZone alloc] initWithName: name data: data]; } } DESTROY(self); return (GSPlaceholderTimeZone*)zone; } - (oneway void) release { return; // placeholders never get released. } - (id) retain { return self; // placeholders never get retained. } @end @implementation NSLocalTimeZone - (NSString*) abbreviation { return [[NSTimeZoneClass defaultTimeZone] abbreviation]; } - (NSString*) abbreviationForDate: (NSDate*)aDate { return [[NSTimeZoneClass defaultTimeZone] abbreviationForDate: aDate]; } - (id) autorelease { return self; } - (id) copy { return self; } - (id) copyWithZone: (NSZone*)z { return self; } - (NSData*) data { return [[NSTimeZoneClass defaultTimeZone] data]; } - (void) encodeWithCoder: (NSCoder*)aCoder { [aCoder encodeObject: @"NSLocalTimeZone"]; } - (id) init { return self; } - (BOOL) isDaylightSavingTime { return [[NSTimeZoneClass defaultTimeZone] isDaylightSavingTime]; } - (BOOL) isDaylightSavingTimeForDate: (NSDate*)aDate { return [[NSTimeZoneClass defaultTimeZone] isDaylightSavingTimeForDate: aDate]; } - (NSString*) name { return [[NSTimeZoneClass defaultTimeZone] name]; } - (oneway void) release { } - (id) retain { return self; } - (NSInteger) secondsFromGMT { return [[NSTimeZoneClass defaultTimeZone] secondsFromGMT]; } - (NSInteger) secondsFromGMTForDate: (NSDate*)aDate { return [[NSTimeZoneClass defaultTimeZone] secondsFromGMTForDate: aDate]; } - (NSArray*) timeZoneDetailArray { return [[NSTimeZoneClass defaultTimeZone] timeZoneDetailArray]; } - (NSTimeZoneDetail*) timeZoneDetailForDate: (NSDate*)date { return [[NSTimeZoneClass defaultTimeZone] timeZoneDetailForDate: date]; } - (NSString*) timeZoneName { return [[NSTimeZoneClass defaultTimeZone] timeZoneName]; } @end @implementation GSAbsTimeZone static int uninitialisedOffset = 100000; - (NSString*) abbreviationForDate: (NSDate*)aDate { return name; } - (void) dealloc { if (offset != uninitialisedOffset) { GS_MUTEX_LOCK(zone_mutex); NSMapRemove(absolutes, (void*)(uintptr_t)offset); GS_MUTEX_UNLOCK(zone_mutex); } RELEASE(name); DEALLOC; } - (void) encodeWithCoder: (NSCoder*)aCoder { [aCoder encodeObject: name]; } - (id) initWithOffset: (NSInteger)anOffset name: (NSString*)aName { GSAbsTimeZone *z; int extra; int sign = anOffset >= 0 ? 1 : -1; /* * Set the uninitialised offset so that dealloc before full * initialisation won't remove the timezeone for offset 0 from cache. */ offset = uninitialisedOffset; /* * Round the offset to the nearest minute, (for MacOS-X compatibility) * and ensure it is no more than 18 hours. */ anOffset *= sign; extra = anOffset % 60; if (extra < 30) { anOffset -= extra; } else { anOffset += 60 - extra; } if (anOffset > 64800) { DESTROY(self); return nil; } anOffset *= sign; GS_MUTEX_LOCK(zone_mutex); if (anOffset % 900 == 0) { z = commonAbsolutes[anOffset/900 + 72]; } else { z = (GSAbsTimeZone*)NSMapGet(absolutes, (void*)(uintptr_t)anOffset); } if (z) { IF_NO_ARC(RETAIN(z);) DESTROY(self); GS_MUTEX_UNLOCK(zone_mutex); return z; } if (aName == nil) { if (anOffset % 60 == 0) { char s = (anOffset >= 0) ? '+' : '-'; unsigned i = (anOffset >= 0) ? anOffset / 60 : -anOffset / 60; unsigned h = (i / 60) % 24; unsigned m = i % 60; char buf[9]; snprintf(buf, sizeof(buf), "GMT%c%02u%02u", s, h, m); name = [[NSString alloc] initWithUTF8String: buf]; } else { /* * Should never happen now we round to the minute * for MacOS-X compatibnility. */ name = [[NSString alloc] initWithFormat: @"NSAbsoluteTimeZone:%"PRIdPTR, anOffset]; } } else { name = [aName copy]; } z = self; offset = anOffset; NSMapInsert(absolutes, (void*)(uintptr_t)anOffset, (void*)z); [zoneDictionary setObject: z forKey: name]; if (anOffset % 900 == 0) { int index = anOffset/900 + 72; if (nil == commonAbsolutes[index]) { commonAbsolutes[index] = RETAIN(z); } } GS_MUTEX_UNLOCK(zone_mutex); return z; } - (BOOL) isDaylightSavingTimeZoneForDate: (NSDate*)aDate { return NO; } - (NSString*) name { return name; } - (NSInteger) secondsFromGMTForDate: (NSDate*)aDate { return offset; } - (NSArray*) timeZoneDetailArray { return [NSArray arrayWithObject: [self timeZoneDetailForDate: nil]]; } - (NSTimeZoneDetail*) timeZoneDetailForDate: (NSDate*)date { return AUTORELEASE([[GSAbsTimeZoneDetail alloc] initWithTimeZone: self]); } - (NSString*) timeZoneName { return name; } @end @implementation GSTimeZoneDetail - (void) dealloc { RELEASE(abbrev); RELEASE(timeZone); DEALLOC } - (id) initWithCoder: (NSCoder*)aDecoder { [aDecoder decodeValueOfObjCType: @encode(id) at: &abbrev]; [aDecoder decodeValueOfObjCType: @encode(int) at: &offset]; [aDecoder decodeValueOfObjCType: @encode(BOOL) at: &is_dst]; return self; } - (id) initWithTimeZone: (NSTimeZone*)aZone withAbbrev: (NSString*)anAbbrev withOffset: (NSInteger)anOffset withDST: (BOOL)isDST { timeZone = RETAIN(aZone); abbrev = RETAIN(anAbbrev); offset = anOffset; is_dst = isDST; return self; } - (BOOL) isDaylightSavingTimeZone { return is_dst; } - (NSString*) name { return [timeZone name]; } - (NSString*) timeZoneAbbreviation { return abbrev; } - (NSArray*) timeZoneDetailArray { return [timeZone timeZoneDetailArray]; } - (NSTimeZoneDetail*) timeZoneDetailForDate: (NSDate*)date { return [timeZone timeZoneDetailForDate: date]; } - (NSInteger) timeZoneSecondsFromGMT { return offset; } - (NSInteger) timeZoneSecondsFromGMTForDate: (NSDate*)aDate { return offset; } @end @implementation GSAbsTimeZoneDetail - (NSString*) abbreviation { return zone->name; } - (NSString*) abbreviationForDate: (NSDate*)aDate { return zone->name; } - (void) dealloc { RELEASE(zone); DEALLOC } - (id) initWithTimeZone: (GSAbsTimeZone*)aZone { zone = RETAIN(aZone); return self; } - (BOOL) isDaylightSavingTimeZone { return NO; } - (BOOL) isDaylightSavingTimeZoneForDate: (NSDate*)aDate { return NO; } - (NSString*) name { return zone->name; } - (NSString*) timeZoneAbbreviation { return zone->name; } - (NSArray*) timeZoneDetailArray { return [zone timeZoneDetailArray]; } - (NSTimeZoneDetail*) timeZoneDetailForDate: (NSDate*)date { return self; } - (NSInteger) timeZoneSecondsFromGMT { return zone->offset; } - (NSInteger) timeZoneSecondsFromGMTForDate: (NSDate*)aDate { return zone->offset; } @end /** *

* The local time zone is obtained from, in order of preference:
* 1) the user defaults database: NSGlobalDomain "Local Time Zone"
* 2) the GNUSTEP_TZ environment variable
* 3) the file "localtime" in System/Library/Libraries/Resources/NSTimeZone
* 4) the TZ environment variable
* 5) The system zone settings (typically in /etc/localtime)
* 6) tzset and tznam on platforms which have it
* 7) Windows registry, on Win32 systems
* 8) or the fallback time zone (which is UTC)
*

*

If the GNUstep time zone datafiles become too out of date, one * can download an updated database from ftp://elsie.nci.nih.gov/pub/ * and compile it as specified in the README file in the * NSTimeZones directory. *

*

Time zone names in NSDates should be GMT, MET etc. not * Europe/Berlin, America/Washington etc. *

*

The problem with this is that various time zones may use the * same abbreviation (e.g. Australia/Brisbane and * America/New_York both use EST), and some time zones * may have different rules for daylight saving time even if the * abbreviation and offsets from UTC are the same. *

*

The problems with depending on the OS for providing time zone * info are that time zone names may vary * wildly between OSes (this could be a big problem when * archiving is used between different systems). *

*

Win32: Time zone names read from the registry are different * from other GNUstep installations. Be careful when moving data * between platforms in this case. *

*/ @implementation NSTimeZone /** * Returns a dictionary containing time zone abbreviations and their * corresponding time zone names. More than one time zone may be associated * with a single abbreviation. In this case, the dictionary contains only * one (usually the most common) time zone name for the abbreviation. */ + (NSDictionary*) abbreviationDictionary { if (abbreviationDictionary != nil) { return abbreviationDictionary; } GS_MUTEX_LOCK(zone_mutex); if (abbreviationDictionary == nil) { NSAutoreleasePool *pool = [NSAutoreleasePool new]; NSString *path; path = _time_zone_path(ABBREV_DICT, @"plist"); if (path != nil) { /* * Fast mechanism ... load prebuilt data from file so we don't * need to load in all time zones. */ abbreviationDictionary = RETAIN([[NSString stringWithContentsOfFile: path] propertyList]); } if (abbreviationDictionary == nil) { NSMutableDictionary *md; NSString *name; NSEnumerator *names; /* * Slow fallback ... load all time zones and generate * abbreviation dictionary from them. */ md = [[NSMutableDictionary alloc] init]; names = [[NSTimeZone knownTimeZoneNames] objectEnumerator]; while ((name = [names nextObject]) != nil) { NSTimeZone *zone; if ((zone = [NSTimeZone timeZoneWithName: name])) { NSEnumerator *details; NSTimeZoneDetail *detail; details = [[zone timeZoneDetailArray] objectEnumerator]; while ((detail = [details nextObject]) != nil) { [md setObject: name forKey: [detail timeZoneAbbreviation]]; } } } if ([md makeImmutable] == YES) { abbreviationDictionary = md; } else { abbreviationDictionary = [md copy]; RELEASE(md); } } [pool drain]; } GS_MUTEX_UNLOCK(zone_mutex); return abbreviationDictionary; } /** * Returns a dictionary that maps abbreviations to the array * containing all the time zone names that use the abbreviation. */ + (NSDictionary*) abbreviationMap { /* Instead of creating the abbreviation dictionary when the class is initialized, we create it when we first need it, since the dictionary can be potentially very large, considering that it's almost never used. */ if (abbreviationMap != nil) { return abbreviationMap; } GS_MUTEX_LOCK(zone_mutex); if (abbreviationMap == nil) { NSAutoreleasePool *pool = [NSAutoreleasePool new]; NSMutableDictionary *md; NSMutableArray *ma; NSString *the_name; NSString *the_abbrev; FILE *file; char abbrev[80]; char name[80]; NSString *path; /* * Read dictionary from file... fast mechanism because we don't have * to create all timezoneas and parse all their data files. */ md = [NSMutableDictionary dictionaryWithCapacity: 100]; path = _time_zone_path (ABBREV_MAP, nil); if (path != nil) { #if defined(_WIN32) unichar mode[3]; mode[0] = 'r'; mode[1] = 'b'; mode[2] = '\0'; file = _wfopen((const unichar*)[path fileSystemRepresentation], mode); #else file = fopen([path fileSystemRepresentation], "r"); #endif if (file == NULL) { GS_MUTEX_UNLOCK(zone_mutex); [NSException raise: NSInternalInconsistencyException format: @"Failed to open time zone abbreviation map."]; } while (fscanf(file, "%79s %79s", abbrev, name) == 2) { the_name = [[NSString alloc] initWithUTF8String: name]; the_abbrev = [[NSString alloc] initWithUTF8String: abbrev]; ma = [md objectForKey: the_abbrev]; if (ma == nil) { ma = [[NSMutableArray alloc] initWithCapacity: 1]; [md setObject: ma forKey: the_abbrev]; RELEASE(ma); } RELEASE(the_abbrev); if ([ma containsObject: the_name] == NO) { [ma addObject: the_name]; } RELEASE(the_name); } fclose(file); } else { NSString *name; NSEnumerator *names; /* * Slow fallback mechanism ... go through all time names * so we load all the time zone data and generate the info * we need from it. */ names = [[NSTimeZone knownTimeZoneNames] objectEnumerator]; while ((name = [names nextObject]) != nil) { NSTimeZone *zone; if ((zone = [NSTimeZone timeZoneWithName: name]) != nil) { NSEnumerator *details; NSTimeZoneDetail *detail; details = [[zone timeZoneDetailArray] objectEnumerator]; while ((detail = [details nextObject]) != nil) { the_abbrev = [detail timeZoneAbbreviation]; ma = [md objectForKey: the_abbrev]; if (ma == nil) { ma = [[NSMutableArray alloc] initWithCapacity: 1]; [md setObject: ma forKey: the_abbrev]; RELEASE(ma); } if ([ma containsObject: name] == NO) { [ma addObject: name]; } } } } } /* Special case: Add the system time zone if * it doesn't exist in the map */ the_abbrev = [systemTimeZone abbreviation]; ma = [md objectForKey: the_abbrev]; if (ma == nil) { ma = [NSMutableArray new]; [md setObject: ma forKey: the_abbrev]; RELEASE(ma); } the_name = [systemTimeZone timeZoneName]; if ([ma containsObject: the_name] == NO) { [ma addObject: the_name]; } if ([md makeImmutable] == YES) { abbreviationMap = RETAIN(md); } else { abbreviationMap = [md copy]; } [pool drain]; } GS_MUTEX_UNLOCK(zone_mutex); return abbreviationMap; } /** * Returns an array of all known time zone names. */ + (NSArray*) knownTimeZoneNames { /* We create the array only when we need it to reduce overhead. */ if (namesArray != nil) { return namesArray; } GS_MUTEX_LOCK(zone_mutex); if (namesArray == nil && NO == [NSObject isExiting]) { unsigned i; NSMutableArray *ma; NSArray *regionsArray; ma = [NSMutableArray new]; regionsArray = [self timeZoneArray]; for (i = 0; i < [regionsArray count]; i++) { NSArray *names = [regionsArray objectAtIndex: i]; [ma addObjectsFromArray: names]; } if ([ma makeImmutable] == YES) { namesArray = ma; } else { namesArray = [ma copy]; RELEASE(ma); } } GS_MUTEX_UNLOCK(zone_mutex); return namesArray; } + (id) allocWithZone: (NSZone*)z { if (self == NSTimeZoneClass) { /* * We return a placeholder object that can * be converted to a real object when its initialisation method * is called. */ if (z == NSDefaultMallocZone() || z == 0) { /* * As a special case, we can return a placeholder for a time zone * in the default zone extremely efficiently. */ return defaultPlaceholderTimeZone; } else { id obj; /* * For anything other than the default zone, we need to * locate the correct placeholder in the (lock protected) * table of placeholders. */ GS_MUTEX_LOCK(zone_mutex); obj = (id)NSMapGet(placeholderMap, (void*)z); if (obj == nil) { /* * There is no placeholder object for this zone, so we * create a new one and use that. */ obj = (id)NSAllocateObject(GSPlaceholderTimeZoneClass, 0, z); NSMapInsert(placeholderMap, (void*)z, (void*)obj); } GS_MUTEX_UNLOCK(zone_mutex); return obj; } } else { return NSAllocateObject(self, 0, z); } } + (void) atExit { id o; /* Deallocate all the placeholders in the map before destroying it. */ GS_MUTEX_LOCK(zone_mutex); if (placeholderMap) { NSMapEnumerator mEnum = NSEnumerateMapTable(placeholderMap); Class c; id o; while (NSNextMapEnumeratorPair(&mEnum, (void *)&c, (void *)&o)) { NSDeallocateObject(o); } NSEndMapTableEnumeration(&mEnum); DESTROY(placeholderMap); } GS_MUTEX_UNLOCK(zone_mutex); DESTROY(zoneDictionary); DESTROY(localTimeZone); DESTROY(defaultTimeZone); DESTROY(systemTimeZone); DESTROY(abbreviationDictionary); DESTROY(abbreviationMap); DESTROY(absolutes); DESTROY(namesArray); DESTROY(regionsArray); o = defaultPlaceholderTimeZone; defaultPlaceholderTimeZone = nil; NSDeallocateObject(o); } /** * Return the default time zone for this process. */ + (NSTimeZone*) defaultTimeZone { NSTimeZone *zone; GS_MUTEX_LOCK(zone_mutex); if (defaultTimeZone == nil) { zone = [self systemTimeZone]; } else { zone = AUTORELEASE(RETAIN(defaultTimeZone)); } GS_MUTEX_UNLOCK(zone_mutex); return zone; } + (void) initialize { static BOOL beenHere = NO; if (NO == beenHere && self == [NSTimeZone class]) { beenHere = YES; NSTimeZoneClass = self; GS_MUTEX_INIT_RECURSIVE(zone_mutex); GSPlaceholderTimeZoneClass = [GSPlaceholderTimeZone class]; zoneDictionary = [[NSMutableDictionary alloc] init]; /* Set up infrastructure for placeholder timezones. */ defaultPlaceholderTimeZone = (GSPlaceholderTimeZone*) NSAllocateObject(GSPlaceholderTimeZoneClass, 0, NSDefaultMallocZone()); placeholderMap = NSCreateMapTable(NSNonOwnedPointerMapKeyCallBacks, NSNonRetainedObjectMapValueCallBacks, 0); /* Hash table lookup for absolute time zones. */ absolutes = NSCreateMapTable(NSIntegerMapKeyCallBacks, NSNonOwnedPointerMapValueCallBacks, 0); localTimeZone = [[NSLocalTimeZone alloc] init]; [[NSNotificationCenter defaultCenter] addObserver: self selector: @selector(_notified:) name: NSUserDefaultsDidChangeNotification object: nil]; [self registerAtExit]; } } /** * Return a proxy to the default time zone for this process. */ + (NSTimeZone*) localTimeZone { return localTimeZone; } /** * Destroy the system time zone so that it will be recreated * next time it is used. */ + (void) resetSystemTimeZone { GS_MUTEX_LOCK(zone_mutex); DESTROY(systemTimeZone); GS_MUTEX_UNLOCK(zone_mutex); [[NSNotificationCenter defaultCenter] postNotificationName: NSSystemTimeZoneDidChangeNotification object: nil]; } /** * Set the default time zone to be used for this process. */ + (void) setDefaultTimeZone: (NSTimeZone*)aTimeZone { if (aTimeZone != defaultTimeZone) { /* * We can't make the localTimeZone the default since that would * cause recursion ... */ if (aTimeZone == localTimeZone) { aTimeZone = [self systemTimeZone]; } GS_MUTEX_LOCK(zone_mutex); ASSIGN(defaultTimeZone, aTimeZone); GS_MUTEX_UNLOCK(zone_mutex); } } /** * Returns the current system time zone for the process. */ + (NSTimeZone*) systemTimeZone { NSTimeZone *zone = nil; GS_MUTEX_LOCK(zone_mutex); if (systemTimeZone == nil) { NSFileManager *dflt = [NSFileManager defaultManager]; NSString *localZoneString = nil; NSString *localZoneSource = nil; /* * setup default value in case something goes wrong. */ systemTimeZone = RETAIN([NSTimeZoneClass timeZoneForSecondsFromGMT: 0]); /* * Try to get timezone from user defaults database */ localZoneSource = [NSString stringWithFormat: @"NSUserDefaults: '%@'", LOCALDBKEY]; localZoneString = [[NSUserDefaults standardUserDefaults] stringForKey: LOCALDBKEY]; /* * Try to get timezone from GNUSTEP_TZ environment variable. */ if (localZoneString == nil) { localZoneSource = _(@"environment variable: 'GNUSTEP_TZ'"); localZoneString = [[[NSProcessInfo processInfo] environment] objectForKey: @"GNUSTEP_TZ"]; } /* * Try to get timezone from LOCAL_TIME_FILE. */ if (localZoneString == nil) { NSString *f = _time_zone_path(LOCAL_TIME_FILE, nil); localZoneSource = [NSString stringWithFormat: @"file: '%@'", f]; if (f != nil) { localZoneString = [NSString stringWithContentsOfFile: f]; localZoneString = [localZoneString stringByTrimmingSpaces]; } } #if defined(_WIN32) /* * Try to get timezone from windows system call. */ { TIME_ZONE_INFORMATION tz; DWORD dst; wchar_t *tzName; #if defined(_MSC_VER) && defined(UCAL_H) // Get time zone name for US locale as expected by ICU method below LANGID origLangID = GetThreadUILanguage(); SetThreadUILanguage(MAKELANGID(LANG_ENGLISH, SUBLANG_ENGLISH_US)); dst = GetTimeZoneInformation(&tz); SetThreadUILanguage(origLangID); // Only tz.StandardName time zone conversions are supported, as // the Zone-Tzid table lacks all daylight time conversions: // e.g. 'W. Europe Daylight Time' <-> 'Europe/Berlin' is not listed. // // See: https://unicode-org.github.io/cldr-staging/charts/latest/supplemental/zone_tzid.html tzName = tz.StandardName; #else dst = GetTimeZoneInformation(&tz); if (dst == TIME_ZONE_ID_DAYLIGHT) tzName = tz.DaylightName; else tzName = tz.StandardName; #endif localZoneSource = @"function: 'GetTimeZoneInformation()'"; #if defined(_MSC_VER) && defined(UCAL_H) // Convert Windows timezone name to IANA identifier if (tzName) { UErrorCode status = U_ZERO_ERROR; UChar ianaTzName[128]; int32_t ianaTzNameLen = ucal_getTimeZoneIDForWindowsID(tzName, -1, NULL, ianaTzName, 128, &status); if (U_SUCCESS(status) && ianaTzNameLen > 0) { localZoneString = [NSString stringWithCharacters: ianaTzName length: ianaTzNameLen]; } else if (U_SUCCESS(status)) { // this happens when ICU has no mapping for the time zone NSLog(@"Unable to map timezone '%ls' to IANA format", tzName); } else { NSLog(@"Error converting timezone '%ls' to IANA format: %s", tzName, u_errorName(status)); } } #endif if (localZoneString == nil) { localZoneString = [NSString stringWithCharacters: tzName length: wcslen(tzName)]; } } #endif if (localZoneString == nil) { if (YES == [dflt isReadableFileAtPath: @"/etc/timezone"]) { NSString *s; s = [NSString stringWithContentsOfFile: @"/etc/timezone"]; s = [s stringByTrimmingSpaces]; if (0 != [s length]) { localZoneSource = _(@"/etc/timezone file"); localZoneString = s; } } } if (localZoneString == nil) { if (YES == [dflt isReadableFileAtPath: @"/etc/sysconfig/clock"]) { NSString *s; NSEnumerator *e; s = [NSString stringWithContentsOfFile: @"/etc/sysconfig/clock"]; e = [[s componentsSeparatedByString: @"\n"] objectEnumerator]; while (nil != (s = [e nextObject])) { s = [s stringByTrimmingSpaces]; // OpenSuse uses the non-standard key TIMEZONE if ([s hasPrefix: @"ZONE"] || [s hasPrefix: @"TIMEZONE"]) { if ([s hasPrefix: @"ZONE"]) s = [s substringFromIndex: 4]; else s = [s substringFromIndex: 8]; s = [s stringByTrimmingSpaces]; if ([s hasPrefix: @"="]) { s = [s substringFromIndex: 1]; s = [s stringByTrimmingSpaces]; if ([s hasPrefix: @"\""]) { s = [s substringFromIndex: 1]; } if ([s hasSuffix: @"\""]) { s = [s substringToIndex: [s length] - 1]; } s = [s stringByTrimmingSpaces]; if ([s length] > 0) { localZoneSource = _(@"/etc/sysconfig/clock file"); localZoneString = s; } } } } } } if (localZoneString == nil) { /* Get the zone name from the localtime file, assuming the file is a symlink to the time zone. Getting the actual data (which is easier) doesn't help, since we won't know the name itself. */ #if defined(HAVE_TZHEAD) && defined(TZDEFAULT) tzdir = RETAIN([NSString stringWithUTF8String: TZDIR]); localZoneString = [NSString stringWithUTF8String: TZDEFAULT]; localZoneSource = [NSString stringWithFormat: @"file (TZDEFAULT): '%@'", localZoneString]; localZoneString = [localZoneString stringByResolvingSymlinksInPath]; #else if ([dflt fileExistsAtPath: SYSTEM_TIME_FILE]) { localZoneString = SYSTEM_TIME_FILE; localZoneSource = [NSString stringWithFormat: @"file (SYSTEM_TIME_FILE): '%@'", localZoneString]; localZoneString = [localZoneString stringByResolvingSymlinksInPath]; /* Guess what tzdir is */ tzdir = [localZoneString stringByDeletingLastPathComponent]; while ([tzdir length] > 2 && [dflt fileExistsAtPath: [tzdir stringByAppendingPathComponent: @"GMT"]] == NO) { tzdir = [tzdir stringByDeletingLastPathComponent]; } if ([tzdir length] <= 2) { localZoneString = tzdir = nil; } else { [tzdir retain]; } } #endif if (localZoneString != nil && [localZoneString hasPrefix: tzdir]) { /* This must be the time zone name */ localZoneString = AUTORELEASE([localZoneString mutableCopy]); [(NSMutableString*)localZoneString deleteCharactersInRange: NSMakeRange(0, [tzdir length])]; while ([localZoneString hasPrefix: @"/"]) { [(NSMutableString*)localZoneString deleteCharactersInRange: NSMakeRange(0, 1)]; } } else { localZoneString = nil; } } /* Try to get timezone from standard unix environment variable. * This is often an ambiguous abbreviation :-( */ if (localZoneString == nil) { localZoneSource = _(@"environment variable: 'TZ'"); localZoneString = [[[NSProcessInfo processInfo] environment] objectForKey: @"TZ"]; } #if HAVE_TZSET && !defined(__FreeBSD__) && !defined(__OpenBSD__) /* * Try to get timezone from tzset and tzname/daylight. * If daylight is non-zero, then tzname[0] is only the name * the the zone for part of the year, so we can't use it as * the definitive zone. * * FreeBSD doesn't implement TZSet fully, so we can't use it there. * Apparently, OpenBSD neither. */ if (localZoneString == nil) { localZoneSource = @"function: 'tzset()/tzname'"; tzset(); if (NULL != tzname[0] && '\0' != *tzname[0] && 0 == daylight) localZoneString = [NSString stringWithUTF8String: tzname[0]]; } #endif if (localZoneString != nil) { NSDebugLLog (@"NSTimeZone", @"Using zone %@", localZoneString); zone = AUTORELEASE([defaultPlaceholderTimeZone initWithName: localZoneString]); if (zone == nil) { NSArray *possibleZoneNames; /* It is not guaranteed on some systems (e.g., Ubuntu) that SYSTEM_TIME_FILE is a symlink. This file is more probably a copy of a zoneinfo file. The above time zone detecting approach can lead to the situation when we can only know about the time zone abbreviation (localZoneString) and (for some time zone abbreviations) the corresponding list of possible time zone names (e.g. SAMT is valid for Pacific/Samoa, Pacific/Pago_Pago, Pacific/Apia, Asia/Samarkand, Europe/Samara, US/Samoa). In such a case the time zone can be selected from the list by comparing the content of SYSTEM_TIME_FILE and the content of zoneinfo files corresponding to the items of that list. */ possibleZoneNames = [[self abbreviationMap] objectForKey: localZoneString]; if (possibleZoneNames != nil) { NSEnumerator *en = [possibleZoneNames objectEnumerator]; NSString *zoneName; NSFileManager *dflt = [NSFileManager defaultManager]; while ((zoneName = [en nextObject]) != nil) { NSString *fileName = [self _getTimeZoneFile: zoneName]; if (fileName != nil && [dflt contentsEqualAtPath: fileName andPath: SYSTEM_TIME_FILE]) { zone = [self timeZoneWithName: zoneName]; if (zone != nil) { GSPrintf(stderr, @"\nIt seems that your operating system does not have a valid timezone name\n" @"configured and is using an abbreviation instead. By comparing timezone\n" @"file data it is has been possible to find the actual timezone used, but\n" @"doing that is a slow process.\n" @"\nYou can avoid slowness of this time zone detecting approach\n" @"by setting the environment variable TZ='%@'\n" @"Or You can override the timezone name by setting the '%@'\n" @"NSUserDefault via the 'defaults' command line utility, a Preferences\n" @"application, or some other utility.\n" @"eg \"defaults write NSGlobalDomain '%@' '%@'\"\n\n", zoneName, LOCALDBKEY, LOCALDBKEY, zoneName); break; } } } } } if (zone == nil) { if (zone == nil) { GSPrintf(stderr, @"\nUnable to create time zone for name: '%@'\n" @"(source '%@').\n", localZoneString, localZoneSource); } if ([localZoneSource hasPrefix: @"file"] || [localZoneSource hasPrefix: @"function"]) { GSPrintf(stderr, @"\nIt seems that your operating system does not have a valid timezone name\n" @"configured (it could be that some other software has set a, possibly\n" @"ambiguous, timezone abbreviation rather than a name) ... please correct\n" @"that or override by setting a timezone name (such as 'Europe/London'\n" @"or 'America/Chicago').\n"); } GSPrintf(stderr, @"\nYou can override the timezone name by setting the '%@'\n" @"NSUserDefault via the 'defaults' command line utility, a Preferences\n" @"application, or some other utility.\n" @"eg \"defaults write NSGlobalDomain '%@' 'Africa/Nairobi'\"\n" @"See '%@'\n" @"for the standard timezones such as 'GB-Eire' or 'America/Chicago'.\n", LOCALDBKEY, LOCALDBKEY, _time_zone_path (ZONES_DIR, nil)); zone = [self timeZoneWithAbbreviation: localZoneString]; if (zone != nil) { NSInteger s; char sign = '+'; s = [zone secondsFromGMT]; if (s < 0) { sign = '-'; s = -s; } GSPrintf(stderr, @"\nSucceeded in treating '%@' as a timezone abbreviation,\n" @"but abbreviations do not uniquely represent timezones, so this may\n" @"not have found the timezone you were expecting. The timezone found\n" @"was '%@' (currently UTC%c%02d%02d)\n\n", localZoneString, [zone name], sign, s/3600, (s/60)%60); } } } else { NSLog(@"No local time zone specified."); } /* * If local time zone fails to allocate, then allocate something * that is sure to succeed (unless we run out of memory, of * course). */ if (zone == nil) { NSLog(@"Using time zone with absolute offset 0."); zone = [self timeZoneForSecondsFromGMT: 0]; } ASSIGN(systemTimeZone, zone); } zone = AUTORELEASE(RETAIN(systemTimeZone)); GS_MUTEX_UNLOCK(zone_mutex); return zone; } /** * Returns an array of all the known regions.
* There are 24 elements, of course, one for each time zone. * Each element contains an array of NSStrings which are * the region names. */ + (NSArray*) timeZoneArray { /* We create the array only when we need it to reduce overhead. */ if (regionsArray != nil) { return regionsArray; } GS_MUTEX_LOCK(zone_mutex); if (regionsArray == nil && NO == [NSObject isExiting]) { NSAutoreleasePool *pool = [NSAutoreleasePool new]; NSMutableArray *temp_array[24]; NSInteger index; NSInteger i; NSString *name; NSString *path; NSString *contents; NSScanner *scanner; NSCharacterSet *newLineSet; NSError *error = nil; for (i = 0; i < 24; i++) { temp_array[i] = [NSMutableArray array]; } path = _time_zone_path(REGIONS_FILE, nil); if (path != nil) { contents = [NSString stringWithContentsOfFile: path encoding: NSUTF8StringEncoding error: &error]; if (!contents) { NSException *exp; NSDictionary *userInfo; GS_MUTEX_UNLOCK(zone_mutex); userInfo = [NSDictionary dictionaryWithObjectsAndKeys: error, @"underlyingError"]; exp = [NSException exceptionWithName: NSInternalInconsistencyException reason: @"Failed to open time zone regions array file." userInfo: userInfo]; [exp raise]; } newLineSet = [NSCharacterSet newlineCharacterSet]; scanner = [NSScanner scannerWithString: contents]; while ([scanner scanInteger: &index] && [scanner scanUpToCharactersFromSet: newLineSet intoString: &name]) { if (index < 0) index = 0; else index %= 24; [temp_array[index] addObject: name]; } } else { NSString *zonedir = [NSTimeZone _getTimeZoneFile: @"WET"]; if (tzdir != nil) { NSFileManager *mgr = [NSFileManager defaultManager]; NSDirectoryEnumerator *enumerator; NSString *name; zonedir = [zonedir stringByDeletingLastPathComponent]; enumerator = [mgr enumeratorAtPath: zonedir]; while ((name = [enumerator nextObject]) != nil) { NSTimeZone *zone = nil; NSString *ext; BOOL isDir; path = [zonedir stringByAppendingPathComponent: name]; ext = [path pathExtension]; if ([mgr fileExistsAtPath: path isDirectory: &isDir] && isDir == NO && [ext isEqual: @"tab"] == NO && [ext isEqual: @"zi"] == NO && [ext isEqual: @"list"] == NO && [ext isEqual: @"leapseconds"] == NO) { zone = [zoneDictionary objectForKey: name]; if (zone == nil) { NSData *data; data = [NSData dataWithContentsOfFile: path]; /* We should really make sure this is a real zone file and not something extra that happens to be in this directory, but initWithName:data: will do this anyway and log a message if not. */ zone = [[self alloc] initWithName: name data: data]; IF_NO_ARC([zone autorelease];) } if (zone != nil) { int offset; NSArray *details; NSTimeZoneDetail *detail; NSEnumerator *e; details = [zone timeZoneDetailArray]; e = [details objectEnumerator]; while ((detail = [e nextObject]) != nil) { if ([detail isDaylightSavingTime] == NO) { break; // Found a standard time } } if (detail == nil && [details count] > 0) { // If no standard time detail = [details objectAtIndex: 0]; } offset = [detail secondsFromGMT]; if (offset < 0) { offset = -offset; offset %= (60 * 60 * 24); if (offset > 0) { offset = -offset; offset += (60 * 60 * 24); } } else { offset %= (60 * 60 * 24); } offset /= (60 * 60); [temp_array[offset] addObject: name]; } } } } } regionsArray = [[NSArray alloc] initWithObjects: temp_array count: 24]; [pool drain]; } GS_MUTEX_UNLOCK(zone_mutex); return regionsArray; } /** * Return a timezone for the specified offset from GMT.
* The timezone returned does not use daylight savings time. * The actual timezone returned has an offset rounded to the nearest * minute.
* Time zones with an offset of more than +/- 18 hours are disallowed, * and nil is returned. */ + (NSTimeZone*) timeZoneForSecondsFromGMT: (NSInteger)seconds { NSTimeZone *zone; int sign = seconds >= 0 ? 1 : -1; int extra; /* * Round the offset to the nearest minute, (for MacOS-X compatibility) * and ensure it is no more than 18 hours. */ seconds *= sign; extra = seconds % 60; if (extra < 30) { seconds -= extra; } else { seconds += 60 - extra; } if (seconds > 64800) { return nil; } seconds *= sign; if (seconds % 900 == 0) { zone = commonAbsolutes[seconds/900 + 72]; } else { GS_MUTEX_LOCK(zone_mutex); zone = (NSTimeZone*)NSMapGet(absolutes, (void*)(uintptr_t)seconds); GS_MUTEX_UNLOCK(zone_mutex); } if (nil == zone) { zone = [[GSAbsTimeZone alloc] initWithOffset: seconds name: nil]; zone = AUTORELEASE(zone); } return zone; } /** * Returns a timezone for the specified abbreviation. The same abbreviations * are used in different regions so this isn't particularly useful.
* Calls NSTimeZone-abbreviation dictionary an so uses a lot of memory. */ + (NSTimeZone*) timeZoneWithAbbreviation: (NSString*)abbreviation { NSTimeZone *zone; NSString *name; name = [[self abbreviationDictionary] objectForKey: abbreviation]; if (name == nil) { zone = nil; } else { zone = [self timeZoneWithName: name data: nil]; } return zone; } /** * Returns a timezone for the specified name. */ + (NSTimeZone*) timeZoneWithName: (NSString*)aTimeZoneName { NSTimeZone *zone; zone = [defaultPlaceholderTimeZone initWithName: aTimeZoneName data: nil]; return AUTORELEASE(zone); } /** * Returns a timezone for aTimeZoneName, created from the supplied * time zone data. Data must be in TZ format as per the Olson database. */ + (NSTimeZone*) timeZoneWithName: (NSString*)name data: (NSData*)data { NSTimeZone *zone; zone = [defaultPlaceholderTimeZone initWithName: name data: data]; return AUTORELEASE(zone); } /** * Returns the abbreviation for this timezone now. * Invokes -abbreviationForDate: */ - (NSString*) abbreviation { return [self abbreviationForDate: [NSDate date]]; } /** * Returns the abbreviation for this timezone at aDate. This may differ * depending on whether daylight savings time is in effect or not. */ - (NSString*) abbreviationForDate: (NSDate*)aDate { NSTimeZoneDetail *detail; NSString *abbr; detail = [self timeZoneDetailForDate: aDate]; abbr = [detail timeZoneAbbreviation]; return abbr; } /** * Returns the Class for this object */ - (Class) classForCoder { return NSTimeZoneClass; } - (id) copyWithZone: (NSZone*)z { return RETAIN(self); } /** * Returns the data with which the receiver was initialised. */ - (NSData*) data { return nil; } /** * Returns the name of this object. */ - (NSString*) description { return [self name]; } - (void) encodeWithCoder: (NSCoder*)aCoder { [aCoder encodeObject: [self name]]; } - (NSUInteger) hash { return [[self name] hash]; } - (id) init { return [self initWithName: @"NSLocalTimeZone" data: nil]; } - (id) initWithCoder: (NSCoder*)aDecoder { NSString *name; name = [aDecoder decodeObject]; self = [self initWithName: name data: nil]; return self; } /** * Initialise a timezone with the supplied name. May return a cached * timezone object rather than the newly created one. */ - (id) initWithName: (NSString*)name { return [self initWithName: name data: nil]; } /** * Initialises a time zone object using the supplied data object.
* This method is intended for internal use by the NSTimeZone * class cluster. * Don't use it ... use -initWithName: instead. */ - (id) initWithName: (NSString*)name data: (NSData*)data { [self notImplemented: _cmd]; return nil; } /** * Returns a boolean indicating whether daylight savings time is in * effect now. Invokes -isDaylightSavingTimeForDate: */ - (BOOL) isDaylightSavingTime { return [self isDaylightSavingTimeForDate: [NSDate date]]; } /** * Returns a boolean indicating whether daylight savings time is in * effect for this time zone at aDate. */ - (BOOL) isDaylightSavingTimeForDate: (NSDate*)aDate { NSTimeZoneDetail *detail; BOOL isDST; detail = [self timeZoneDetailForDate: aDate]; isDST = [detail isDaylightSavingTimeZone]; return isDST; } - (BOOL) isEqual: (id)other { if (other == self) return YES; if ([other isKindOfClass: NSTimeZoneClass] == NO) return NO; return [self isEqualToTimeZone: other]; } /** * Returns YES if the time zones have the same name. */ - (BOOL) isEqualToTimeZone: (NSTimeZone*)aTimeZone { if (aTimeZone == self) return YES; if ([[self name] isEqual: [aTimeZone name]] == NO) return NO; if (([self data] == nil && [aTimeZone data] == nil) || [[self name] isEqual: [aTimeZone name]] == YES) return YES; return NO; } /** * Returns the name of the timezone */ - (NSString*) name { return [self subclassResponsibility: _cmd]; } - (id) replacementObjectForPortCoder: (NSPortCoder*)aCoder { if ([aCoder isByref] == NO) { return self; } return [super replacementObjectForPortCoder: aCoder]; } /** * Returns the number of seconds by which the receiver differs * from Greenwich Mean Time at the current date and time.
* Invokes -secondsFromGMTForDate: */ - (NSInteger) secondsFromGMT { return [self secondsFromGMTForDate: [NSDate date]]; } /** * Returns the number of seconds by which the receiver differs * from Greenwich Mean Time at the date aDate.
* If the time zone uses daylight savings time, the returned value * will vary at different times of year. */ - (NSInteger) secondsFromGMTForDate: (NSDate*)aDate { NSTimeZoneDetail *detail; int offset; detail = [self timeZoneDetailForDate: aDate]; offset = [detail timeZoneSecondsFromGMT]; return offset; } /** * DEPRECATED: see NSTimeZoneDetail */ - (NSArray*) timeZoneDetailArray { return [self subclassResponsibility: _cmd]; } /** * DEPRECATED: see NSTimeZoneDetail */ - (NSTimeZoneDetail*) timeZoneDetailForDate: (NSDate*)date { return [self subclassResponsibility: _cmd]; } /** * Returns the name of this timezone. */ - (NSString*) timeZoneName { return [self name]; } - (NSTimeInterval) daylightSavingTimeOffsetForDate: (NSDate *)aDate { #if GS_USE_ICU == 1 NSTimeInterval result; UCalendar *cal; UErrorCode err = U_ZERO_ERROR; cal = ICUCalendarSetup (self, nil); if (cal == NULL) return 0.0; ucal_setMillis (cal, ([aDate timeIntervalSince1970] * 1000.0), &err); result = (double)ucal_get (cal, UCAL_DST_OFFSET, &err) / 1000.0; if (U_FAILURE(err)) result = 0.0; ucal_close (cal); return result; #else return 0.0; // FIXME #endif } - (NSDate *) nextDaylightSavingTimeTransitionAfterDate: (NSDate *)aDate { #if GS_USE_ICU == 1 /* ICU doesn't provide transition information per se. * The canonical method of retrieving this piece of information is to * use binary search. */ int32_t originalOffset, currentOffset; UCalendar *cal; UErrorCode err = U_ZERO_ERROR; UDate currentTime; int i; NSDate* result = nil; cal = ICUCalendarSetup (self, nil); if (cal == NULL) return nil; currentTime = [aDate timeIntervalSince1970] * 1000.0; ucal_setMillis (cal, currentTime, &err); originalOffset = ucal_get (cal, UCAL_DST_OFFSET, &err); if (U_FAILURE(err)) return nil; /* First try to find the next transition by adding a week at a time */ /* Avoid ending in an infinite loop in case there is no transition at all */ for (i = 0; i < 53; i++) { /* Add a single week */ currentTime += WEEK_MILLISECONDS; ucal_setMillis (cal, currentTime, &err); if (U_FAILURE(err)) break; currentOffset = ucal_get (cal, UCAL_DST_OFFSET, &err); if (U_FAILURE(err)) break; if (currentOffset != originalOffset) { double interval = WEEK_MILLISECONDS / 2.0; /* Now use bisection to determine the exact moment */ while (interval >= 1.0) { ucal_setMillis (cal, currentTime - interval, &err); currentOffset = ucal_get (cal, UCAL_DST_OFFSET, &err); if (currentOffset != originalOffset) currentTime -= interval; /* it is in the lower half */ interval /= 2.0; } result = [NSDate dateWithTimeIntervalSince1970: floor(currentTime/1000.0)]; } } ucal_close (cal); return result; #else return nil; // FIXME; #endif } - (NSTimeInterval) daylightSavingTimeOffset { return [self daylightSavingTimeOffsetForDate: [NSDate date]]; } - (NSDate *) nextDaylightSavingTimeTransition { return [self nextDaylightSavingTimeTransitionAfterDate: [NSDate date]]; } - (NSString *) localizedName: (NSTimeZoneNameStyle)style locale: (NSLocale *)locale { #if GS_USE_ICU == 1 UChar *result; const char *cLocale; int32_t len; UCalendar *cal; UErrorCode err = U_ZERO_ERROR; cal = ICUCalendarSetup(self, locale); if (cal == NULL) return nil; cLocale = [[locale localeIdentifier] UTF8String]; result = NSZoneMalloc([self zone], BUFFER_SIZE * sizeof(UChar)); len = ucal_getTimeZoneDisplayName(cal, _NSToICUTZDisplayStyle(style), cLocale, result, BUFFER_SIZE, &err); if (len > BUFFER_SIZE) { result = NSZoneRealloc([self zone], result, len * sizeof(UChar)); ucal_getTimeZoneDisplayName(cal, _NSToICUTZDisplayStyle(style), cLocale, result, len, &err); } ucal_close(cal); return AUTORELEASE([[NSString alloc] initWithCharactersNoCopy: result length: len freeWhenDone: YES]); #else return nil; // FIXME; #endif } @end /** * This class serves no useful purpose in GNUstep other than to provide * a backup mechanism for handling abbreviations where the precomputed * data files cannot be found. It is provided primarily for backward * compatibility with the OpenStep spec. It is missing entirely from MacOS-X. */ @implementation NSTimeZoneDetail - (NSString*) description { return [NSString stringWithFormat: @"%@(%@, %s%"PRIdPTR")", [self name], [self timeZoneAbbreviation], ([self isDaylightSavingTimeZone]? "IS_DST, ": ""), [self timeZoneSecondsFromGMT]]; } /** * DEPRECATED: Class is no longer used. */ - (BOOL) isDaylightSavingTimeZone { [self subclassResponsibility: _cmd]; return NO; } /** * DEPRECATED: Class is no longer used. */ - (NSString*) timeZoneAbbreviation { return [self subclassResponsibility: _cmd]; } /** * DEPRECATED: Class is no longer used. */ - (NSInteger) timeZoneSecondsFromGMT { [self subclassResponsibility: _cmd]; return 0; } @end @implementation NSTimeZone (Private) /** * Common locations for timezone info on unix systems. */ static NSString *zoneDirs[] = { #ifdef TZDIR @TZDIR, #endif @"/usr/share/zoneinfo", @"/usr/lib/zoneinfo", @"/usr/local/share/zoneinfo", @"/usr/local/lib/zoneinfo", @"/etc/zoneinfo", @"/usr/local/etc/zoneinfo" }; /** * Returns the path to the named zone info file. */ + (NSString*) _getTimeZoneFile: (NSString *)name { static BOOL beenHere = NO; NSString *dir = nil; BOOL isDir; if (beenHere == NO && tzdir == nil) { GS_MUTEX_LOCK(zone_mutex); if (beenHere == NO && tzdir == nil) { NSFileManager *mgr = [NSFileManager defaultManager]; NSString *zonedir = nil; unsigned i; for (i = 0; i < sizeof(zoneDirs)/sizeof(zoneDirs[0]); i++) { BOOL isDir; zonedir = [zoneDirs[i] stringByAppendingPathComponent: POSIX_TZONES]; if ([mgr fileExistsAtPath: zonedir isDirectory: &isDir] && isDir) { tzdir = RETAIN(zonedir); break; // use first one } } beenHere = YES; } GS_MUTEX_UNLOCK(zone_mutex); } /* Use the system zone info if possible, otherwise, use our installed info. */ if (tzdir && [[NSFileManager defaultManager] fileExistsAtPath: [tzdir stringByAppendingPathComponent: name] isDirectory: &isDir] == YES && isDir == NO) { dir = tzdir; } if (dir == nil) { dir = _time_zone_path (ZONES_DIR, nil); } return [dir stringByAppendingPathComponent: name]; } + (void) _notified: (NSNotification*)n { NSString *name; /* If the name of the system time zone has changed ... * get a new system time zone. */ name = [[NSUserDefaults standardUserDefaults] stringForKey: LOCALDBKEY]; if ([name length] > 0 && [name isEqual: [[self systemTimeZone] name]] == NO) { [self resetSystemTimeZone]; [self systemTimeZone]; } } @end #if defined(_WIN32) /* Timezone information data as stored in the registry */ typedef struct TZI_format { LONG Bias; LONG StandardBias; LONG DaylightBias; SYSTEMTIME StandardDate; SYSTEMTIME DaylightDate; } TZI; static inline unsigned int lastDayOfGregorianMonth(int month, int year) { switch (month) { case 2: if ((((year % 4) == 0) && ((year % 100) != 0)) || ((year % 400) == 0)) return 29; else return 28; case 4: case 6: case 9: case 11: return 30; default: return 31; } } /* IMPORT from NSCalendar date */ void GSBreakTime(NSTimeInterval when, NSInteger *year, NSInteger *month, NSInteger *day, NSInteger *hour, NSInteger *minute, NSInteger *second, NSInteger *mil); @implementation GSWindowsTimeZone - (NSString*) abbreviationForDate: (NSDate*)aDate { if ([self isDaylightSavingTimeForDate: aDate]) return daylightZoneNameAbbr; return timeZoneNameAbbr; } - (NSData*) data { return 0; } - (void) dealloc { RELEASE(timeZoneName); RELEASE(daylightZoneName); RELEASE(timeZoneNameAbbr); RELEASE(daylightZoneNameAbbr); DEALLOC } - (id) initWithName: (NSString*)name data: (NSData*)data { HKEY regDirKey; BOOL isNT = NO; BOOL regFound=NO; BOOL tzFound = NO; /* Open the key in the local machine hive where * the time zone data is stored. */ if (ERROR_SUCCESS == RegOpenKeyExW(HKEY_LOCAL_MACHINE, L"SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion\\Time Zones", 0, KEY_READ, ®DirKey)) { isNT = YES; regFound = YES; } else { if (ERROR_SUCCESS == RegOpenKeyExW(HKEY_LOCAL_MACHINE, L"SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Time Zones", 0, KEY_READ, ®DirKey)) { regFound = YES; } } if (regFound) { /* Iterate over all subKeys in the registry to find the right one. Unfortunately name is a localized value. The keys in the registry are unlocalized names. */ wchar_t achKey[255]; // buffer for subkey name DWORD cbName; // size of name string wchar_t achClass[MAX_PATH] = L""; // buffer for class name DWORD cchClassName = MAX_PATH; // size of class string DWORD cSubKeys = 0; // number of subkeys DWORD cbMaxSubKey; // longest subkey size DWORD cchMaxClass; // longest class string DWORD cValues; // number of values for key DWORD cchMaxValue; // longest value name DWORD cbMaxValueData; // longest value data DWORD cbSecurityDescriptor; // size of security descriptor FILETIME ftLastWriteTime; // last write time DWORD i, retCode; /* Get the class name and the value count. */ retCode = RegQueryInfoKeyW( regDirKey, // key handle achClass, // buffer for class name &cchClassName, // size of class string NULL, // reserved &cSubKeys, // number of subkeys &cbMaxSubKey, // longest subkey size &cchMaxClass, // longest class string &cValues, // number of values for this key &cchMaxValue, // longest value name &cbMaxValueData, // longest value data &cbSecurityDescriptor, // security descriptor &ftLastWriteTime); // last write time if (cSubKeys && (retCode == ERROR_SUCCESS)) { unsigned wLen = [name length]; wchar_t *wName = malloc((wLen+1) * sizeof(wchar_t)); if (wName) { [name getCharacters: wName]; wName[wLen] = 0; for (i = 0; i < cSubKeys && !tzFound; i++) { cbName = 255; retCode = RegEnumKeyExW(regDirKey, i, achKey, &cbName, NULL, NULL, NULL, &ftLastWriteTime); if (retCode == ERROR_SUCCESS) { wchar_t keyBuffer[16384]; HKEY regKey; if (isNT) wcscpy(keyBuffer, L"SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion\\Time Zones\\"); else wcscpy(keyBuffer, L"SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Time Zones\\"); wcscat(keyBuffer, achKey); if (ERROR_SUCCESS == RegOpenKeyExW(HKEY_LOCAL_MACHINE, keyBuffer, 0, KEY_READ, ®Key)) { wchar_t buf[256]; wchar_t standardName[256]; wchar_t daylightName[256]; DWORD bufsize; DWORD type; /* check standardname */ standardName[0] = L'\0'; bufsize = sizeof(buf); if (ERROR_SUCCESS == RegQueryValueExW(regKey, L"Std", 0, &type, (BYTE *)buf, &bufsize)) { wcscpy(standardName, buf); if (wcscmp(standardName, wName) == 0) tzFound = YES; } /* check daylightname */ daylightName[0] = L'\0'; bufsize = sizeof(buf); if (ERROR_SUCCESS == RegQueryValueExW(regKey, L"Dlt", 0, &type, (BYTE *)buf, &bufsize)) { wcscpy(daylightName, buf); if (wcscmp(daylightName, wName) == 0) tzFound = YES; } if (tzFound) { /* Read in the time zone data */ bufsize = sizeof(buf); if (ERROR_SUCCESS == RegQueryValueExW(regKey, L"TZI", 0, &type, (BYTE *)buf, &bufsize)) { TZI *tzi = (void*)buf; Bias = tzi->Bias; StandardBias = tzi->StandardBias; DaylightBias = tzi->DaylightBias; StandardDate = tzi->StandardDate; DaylightDate = tzi->DaylightDate; } /* Set the standard name for the time zone. */ if (wcslen(standardName)) { int a, b; ASSIGN(timeZoneName, [NSString stringWithCharacters: standardName length: wcslen(standardName)]); /* Abbr generated here is IMHO * a bit suspicous but I kept it */ for (a = 0, b = 0; standardName[a]; a++) { if (iswupper(standardName[a])) standardName[b++] = standardName[a]; } standardName[b] = L'\0'; ASSIGN(timeZoneNameAbbr, [NSString stringWithCharacters: standardName length: wcslen(standardName)]); } /* Set the daylight savings name * for the time zone. */ if (wcslen(daylightName)) { int a, b; ASSIGN(daylightZoneName, [NSString stringWithCharacters: daylightName length: wcslen(daylightName)]); /* Abbr generated here is IMHO * a bit suspicous but I kept it */ for (a = 0, b = 0; daylightName[a]; a++) { if (iswupper(daylightName[a])) daylightName[b++] = daylightName[a]; } daylightName[b] = L'\0'; ASSIGN(daylightZoneNameAbbr, [NSString stringWithCharacters: daylightName length: wcslen(daylightName)]); } } RegCloseKey(regKey); } } } free(wName); } } RegCloseKey(regDirKey); } if (NO == tzFound) { DESTROY(self); } return self; } - (BOOL) isDaylightSavingTimeForDate: (NSDate*)aDate { NSInteger year, month, day, hour, minute, second, mil; int dow; int daylightdate, count, maxdate; NSTimeInterval when; if (DaylightDate.wMonth == 0) return NO; when = [aDate timeIntervalSinceReferenceDate] - Bias*60; GSBreakTime(when, &year, &month, &day, &hour, &minute, &second, &mil); // Check north globe if (StandardDate.wMonth >= DaylightDate.wMonth) { // Before April or after October is Std if (month < DaylightDate.wMonth || month > StandardDate.wMonth) { return NO; } // After April and before October is DST if (month > DaylightDate.wMonth && month < StandardDate.wMonth) { return YES; } } else { /* check south globe * Before April or after October is DST */ if (month < StandardDate.wMonth || month > DaylightDate.wMonth) { return YES; } // After April and before October is Std if (month > StandardDate.wMonth && month < DaylightDate.wMonth) { return NO; } } dow = ((NSInteger)((when / 86400.0) + GREGORIAN_REFERENCE)) % 7; if (dow < 0) dow += 7; if (month == DaylightDate.wMonth /* April */) { daylightdate = day - dow + DaylightDate.wDayOfWeek; maxdate = lastDayOfGregorianMonth(DaylightDate.wMonth, year)-7; while (daylightdate > 7) daylightdate -= 7; if (daylightdate < 1) daylightdate += 7; count = DaylightDate.wDay; while (count > 1 && daylightdate < maxdate) { daylightdate += 7; count--; } if (day > daylightdate) return YES; if (day < daylightdate) return NO; if (hour > DaylightDate.wHour) return YES; if (hour < DaylightDate.wHour) return NO; if (minute > DaylightDate.wMinute) return YES; if (minute < DaylightDate.wMinute) return NO; if (second > DaylightDate.wSecond) return YES; if (second < DaylightDate.wSecond) return NO; if (mil >= DaylightDate.wMilliseconds) return YES; return NO; } if (month == StandardDate.wMonth /* October */) { daylightdate = day - dow + StandardDate.wDayOfWeek; maxdate = lastDayOfGregorianMonth(StandardDate.wMonth, year)-7; while (daylightdate > 7) daylightdate -= 7; if (daylightdate < 1) daylightdate += 7; count = StandardDate.wDay; while (count > 1 && daylightdate < maxdate) { daylightdate += 7; count--; } if (day > daylightdate) return NO; if (day < daylightdate) return YES; if (hour > StandardDate.wHour) return NO; if (hour < StandardDate.wHour) return YES; if (minute > StandardDate.wMinute) return NO; if (minute < StandardDate.wMinute) return YES; if (second > StandardDate.wSecond) return NO; if (second < StandardDate.wSecond) return YES; if (mil >= StandardDate.wMilliseconds) return NO; return YES; } return NO; // Never reached } - (NSString*) name { TIME_ZONE_INFORMATION tz; DWORD DST = GetTimeZoneInformation(&tz); if (DST == TIME_ZONE_ID_DAYLIGHT) { return daylightZoneName; } else { return timeZoneName; } } - (NSInteger) secondsFromGMTForDate: (NSDate*)aDate { if ([self isDaylightSavingTimeForDate: aDate]) return -Bias*60 - DaylightBias*60; return -Bias*60 - StandardBias*60; } - (NSArray*) timeZoneDetailArray { return [NSArray arrayWithObjects: [[[GSTimeZoneDetail alloc] initWithTimeZone: self withAbbrev: timeZoneNameAbbr withOffset: -Bias*60 - StandardBias*60 withDST: NO] autorelease], [[[GSTimeZoneDetail alloc] initWithTimeZone: self withAbbrev: daylightZoneNameAbbr withOffset: -Bias*60 - DaylightBias*60 withDST: YES] autorelease], 0]; } - (NSTimeZoneDetail*) timeZoneDetailForDate: (NSDate*)aDate { GSTimeZoneDetail *detail; int offset; BOOL isDST = [self isDaylightSavingTimeForDate: aDate]; NSString *abbr; if (isDST) { offset = -Bias*60 - DaylightBias*60; abbr = daylightZoneNameAbbr; } else { offset = -Bias*60 - StandardBias*60; abbr = timeZoneNameAbbr; } detail = [GSTimeZoneDetail alloc]; detail = [detail initWithTimeZone: self withAbbrev: abbr withOffset: offset withDST: isDST]; return detail; } - (NSString*) timeZoneName { return [self name]; } @end #endif // _WIN32 @implementation GSTimeZone static NSTimeZoneDetail* newDetailInZoneForType(GSTimeZone *zone, TypeInfo *type) { GSTimeZoneDetail *detail; NSString *abbr; abbr = [[NSString alloc] initWithUTF8String: type->abbreviation]; detail = [[GSTimeZoneDetail alloc] initWithTimeZone: zone withAbbrev: abbr withOffset: type->offset withDST: type->isdst]; RELEASE(abbr); return detail; } /** * Obtain TypeInfo from transisions or TZstring */ static TypeInfo getTypeInfo(NSTimeInterval since, GSTimeZone *zone) { time_t when = (time_t)since; gstm tm; TypeInfo type; if (localsub(zone->sp, &when, 0, &tm) == NULL) { memset(&type, '\0', sizeof(type)); } else { type.offset = tm.tm_gmtoff; type.isdst = tm.tm_isdst; type.abbr_idx = 0; type.abbreviation = tm.tm_zone; } return type; } - (NSString*) abbreviationForDate: (NSDate*)aDate { TypeInfo type = getTypeInfo([aDate timeIntervalSince1970], self); if (NULL == type.abbreviation) { return nil; } return [NSString stringWithUTF8String: type.abbreviation]; } - (NSData*) data { return timeZoneData; } - (void) dealloc { RELEASE(timeZoneName); RELEASE(timeZoneData); if (sp != 0) { free(sp); } DEALLOC } - (id) initWithName: (NSString*)name data: (NSData*)data { static NSString *fileException = @"GSTimeZoneFileException"; union local_storage *lsp; const char *zoneName; /* The placeholder class should have dealt with loading the data */ NSAssert([data isKindOfClass: [NSData class]], NSInvalidArgumentException); NSAssert([name isKindOfClass: [NSString class]], NSInvalidArgumentException); timeZoneName = [name copy]; timeZoneData = [data copy]; zoneName = [timeZoneName UTF8String]; sp = malloc(sizeof(*sp)); memset((char *)sp, '\0', sizeof(*sp)); NS_DURING { size_t nread; union input_buffer *up; lsp = malloc(sizeof(*lsp)); up = &lsp->u.u; nread = [data length]; if (nread > sizeof(up->buf)) { [NSException raise: fileException format: @"data too large"]; } [data getBytes: up->buf]; if (tzloadbody1(zoneName, sp, true, lsp, nread) != 0) { [NSException raise: fileException format: @"cannot load data"]; } free(lsp); lsp = NULL; scrub_abbrs(sp); } NS_HANDLER { if (lsp) { free(lsp); } DESTROY(self); NSLog(@"Unable to obtain time zone `%@'... %@", name, localException); if ([localException name] != fileException) { [localException raise]; } return nil; } NS_ENDHANDLER GS_MUTEX_LOCK(zone_mutex); [zoneDictionary setObject: self forKey: timeZoneName]; GS_MUTEX_UNLOCK(zone_mutex); return self; } - (BOOL) isDaylightSavingTimeForDate: (NSDate*)aDate { TypeInfo type = getTypeInfo([aDate timeIntervalSince1970], self); return type.isdst; } - (NSString*) name { return timeZoneName; } - (NSInteger) secondsFromGMTForDate: (NSDate*)aDate { TypeInfo type = getTypeInfo([aDate timeIntervalSince1970], self); return type.offset; } - (NSArray*) timeZoneDetailArray { NSTimeZoneDetail *details[sp->typecnt]; unsigned i; NSArray *array; for (i = 0; i < sp->typecnt; i++) { const struct _ttinfo *const ttisp = &sp->ttis[i]; const char *abbr = &sp->chars[ttisp->tt_desigidx]; TypeInfo type; type.offset = ttisp->tt_utoff; type.isdst = ttisp->tt_isdst; type.abbr_idx = ttisp->tt_desigidx; type.abbreviation = abbr; details[i] = newDetailInZoneForType(self, &type); } array = [NSArray arrayWithObjects: details count: sp->typecnt]; for (i = 0; i < sp->typecnt; i++) { RELEASE(details[i]); } return array; } - (NSTimeZoneDetail*) timeZoneDetailForDate: (NSDate*)aDate { TypeInfo type; NSTimeZoneDetail *detail; type = getTypeInfo([aDate timeIntervalSince1970], self); detail = newDetailInZoneForType(self, &type); return AUTORELEASE(detail); } - (NSString*) timeZoneName { return timeZoneName; } @end