2005-04-16 15:20:36 -07:00
/* Copyright (C) 1993, 1994, 1995, 1996, 1997 Free Software Foundation, Inc.
This file is part of the GNU C Library .
Contributed by Paul Eggert ( eggert @ twinsun . com ) .
The GNU C Library is free software ; you can redistribute it and / or
modify it under the terms of the GNU Library General Public License as
published by the Free Software Foundation ; either version 2 of the
License , or ( at your option ) any later version .
The GNU C 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
Library General Public License for more details .
You should have received a copy of the GNU Library General Public
License along with the GNU C Library ; see the file COPYING . LIB . If not ,
write to the Free Software Foundation , Inc . , 59 Temple Place - Suite 330 ,
Boston , MA 02111 - 1307 , USA . */
/*
2008-02-08 04:20:36 -08:00
* dgb 10 / 02 / 98 : ripped this from glibc source to help convert timestamps
* to unix time
* 10 / 04 / 98 : added new table - based lookup after seeing how ugly
* the gnu code is
2005-04-16 15:20:36 -07:00
* blf 09 / 27 / 99 : ripped out all the old code and inserted new table from
2007-07-21 04:37:18 -07:00
* John Brockmeyer ( without leap second corrections )
* rewrote udf_stamp_to_time and fixed timezone accounting in
* udf_time_to_stamp .
2005-04-16 15:20:36 -07:00
*/
/*
* We don ' t take into account leap seconds . This may be correct or incorrect .
* For more NIST information ( especially dealing with leap seconds ) , see :
2007-07-21 04:37:18 -07:00
* http : //www.boulder.nist.gov/timefreq/pubs/bulletin/leapsecond.htm
2005-04-16 15:20:36 -07:00
*/
# include <linux/types.h>
# include <linux/kernel.h>
# include "udfdecl.h"
# define EPOCH_YEAR 1970
# ifndef __isleap
/* Nonzero if YEAR is a leap year (every 4 years,
except every 100 th isn ' t , and every 400 th is ) . */
# define __isleap(year) \
( ( year ) % 4 = = 0 & & ( ( year ) % 100 ! = 0 | | ( year ) % 400 = = 0 ) )
# endif
/* How many days come before each month (0-12). */
2007-07-19 01:47:43 -07:00
static const unsigned short int __mon_yday [ 2 ] [ 13 ] = {
2005-04-16 15:20:36 -07:00
/* Normal years. */
2007-07-19 01:47:43 -07:00
{ 0 , 31 , 59 , 90 , 120 , 151 , 181 , 212 , 243 , 273 , 304 , 334 , 365 } ,
2005-04-16 15:20:36 -07:00
/* Leap years. */
2007-07-19 01:47:43 -07:00
{ 0 , 31 , 60 , 91 , 121 , 152 , 182 , 213 , 244 , 274 , 305 , 335 , 366 }
2005-04-16 15:20:36 -07:00
} ;
# define MAX_YEAR_SECONDS 69
2007-07-21 04:37:18 -07:00
# define SPD 0x15180 /*3600*24 */
2008-02-08 04:20:36 -08:00
# define SPY(y, l, s) (SPD * (365 * y + l) + s)
static time_t year_seconds [ MAX_YEAR_SECONDS ] = {
/*1970*/ SPY ( 0 , 0 , 0 ) , SPY ( 1 , 0 , 0 ) , SPY ( 2 , 0 , 0 ) , SPY ( 3 , 1 , 0 ) ,
/*1974*/ SPY ( 4 , 1 , 0 ) , SPY ( 5 , 1 , 0 ) , SPY ( 6 , 1 , 0 ) , SPY ( 7 , 2 , 0 ) ,
/*1978*/ SPY ( 8 , 2 , 0 ) , SPY ( 9 , 2 , 0 ) , SPY ( 10 , 2 , 0 ) , SPY ( 11 , 3 , 0 ) ,
/*1982*/ SPY ( 12 , 3 , 0 ) , SPY ( 13 , 3 , 0 ) , SPY ( 14 , 3 , 0 ) , SPY ( 15 , 4 , 0 ) ,
/*1986*/ SPY ( 16 , 4 , 0 ) , SPY ( 17 , 4 , 0 ) , SPY ( 18 , 4 , 0 ) , SPY ( 19 , 5 , 0 ) ,
/*1990*/ SPY ( 20 , 5 , 0 ) , SPY ( 21 , 5 , 0 ) , SPY ( 22 , 5 , 0 ) , SPY ( 23 , 6 , 0 ) ,
/*1994*/ SPY ( 24 , 6 , 0 ) , SPY ( 25 , 6 , 0 ) , SPY ( 26 , 6 , 0 ) , SPY ( 27 , 7 , 0 ) ,
/*1998*/ SPY ( 28 , 7 , 0 ) , SPY ( 29 , 7 , 0 ) , SPY ( 30 , 7 , 0 ) , SPY ( 31 , 8 , 0 ) ,
/*2002*/ SPY ( 32 , 8 , 0 ) , SPY ( 33 , 8 , 0 ) , SPY ( 34 , 8 , 0 ) , SPY ( 35 , 9 , 0 ) ,
/*2006*/ SPY ( 36 , 9 , 0 ) , SPY ( 37 , 9 , 0 ) , SPY ( 38 , 9 , 0 ) , SPY ( 39 , 10 , 0 ) ,
/*2010*/ SPY ( 40 , 10 , 0 ) , SPY ( 41 , 10 , 0 ) , SPY ( 42 , 10 , 0 ) , SPY ( 43 , 11 , 0 ) ,
/*2014*/ SPY ( 44 , 11 , 0 ) , SPY ( 45 , 11 , 0 ) , SPY ( 46 , 11 , 0 ) , SPY ( 47 , 12 , 0 ) ,
/*2018*/ SPY ( 48 , 12 , 0 ) , SPY ( 49 , 12 , 0 ) , SPY ( 50 , 12 , 0 ) , SPY ( 51 , 13 , 0 ) ,
/*2022*/ SPY ( 52 , 13 , 0 ) , SPY ( 53 , 13 , 0 ) , SPY ( 54 , 13 , 0 ) , SPY ( 55 , 14 , 0 ) ,
/*2026*/ SPY ( 56 , 14 , 0 ) , SPY ( 57 , 14 , 0 ) , SPY ( 58 , 14 , 0 ) , SPY ( 59 , 15 , 0 ) ,
/*2030*/ SPY ( 60 , 15 , 0 ) , SPY ( 61 , 15 , 0 ) , SPY ( 62 , 15 , 0 ) , SPY ( 63 , 16 , 0 ) ,
/*2034*/ SPY ( 64 , 16 , 0 ) , SPY ( 65 , 16 , 0 ) , SPY ( 66 , 16 , 0 ) , SPY ( 67 , 17 , 0 ) ,
/*2038*/ SPY ( 68 , 17 , 0 )
2005-04-16 15:20:36 -07:00
} ;
extern struct timezone sys_tz ;
# define SECS_PER_HOUR (60 * 60)
# define SECS_PER_DAY (SECS_PER_HOUR * 24)
2008-10-15 12:28:03 +02:00
struct timespec *
udf_disk_stamp_to_time ( struct timespec * dest , struct timestamp src )
2005-04-16 15:20:36 -07:00
{
int yday ;
2008-02-10 11:25:31 +01:00
u16 typeAndTimezone = le16_to_cpu ( src . typeAndTimezone ) ;
u16 year = le16_to_cpu ( src . year ) ;
uint8_t type = typeAndTimezone > > 12 ;
2005-04-16 15:20:36 -07:00
int16_t offset ;
2007-07-19 01:47:43 -07:00
if ( type = = 1 ) {
2008-02-10 11:25:31 +01:00
offset = typeAndTimezone < < 4 ;
2005-04-16 15:20:36 -07:00
/* sign extent offset */
offset = ( offset > > 4 ) ;
2007-07-21 04:37:18 -07:00
if ( offset = = - 2047 ) /* unspecified offset */
2005-04-16 15:20:36 -07:00
offset = 0 ;
2008-02-27 22:50:14 +01:00
} else
2005-04-16 15:20:36 -07:00
offset = 0 ;
2008-02-10 11:25:31 +01:00
if ( ( year < EPOCH_YEAR ) | |
( year > = EPOCH_YEAR + MAX_YEAR_SECONDS ) ) {
2005-04-16 15:20:36 -07:00
return NULL ;
}
2008-02-10 11:25:31 +01:00
dest - > tv_sec = year_seconds [ year - EPOCH_YEAR ] ;
2008-02-27 22:50:14 +01:00
dest - > tv_sec - = offset * 60 ;
2005-04-16 15:20:36 -07:00
2008-02-10 11:25:31 +01:00
yday = ( ( __mon_yday [ __isleap ( year ) ] [ src . month - 1 ] ) + src . day - 1 ) ;
2008-02-27 22:50:14 +01:00
dest - > tv_sec + = ( ( ( yday * 24 ) + src . hour ) * 60 + src . minute ) * 60 + src . second ;
dest - > tv_nsec = 1000 * ( src . centiseconds * 10000 +
src . hundredsOfMicroseconds * 100 + src . microseconds ) ;
2005-04-16 15:20:36 -07:00
return dest ;
}
2008-10-15 12:28:03 +02:00
struct timestamp *
udf_time_to_disk_stamp ( struct timestamp * dest , struct timespec ts )
2005-04-16 15:20:36 -07:00
{
long int days , rem , y ;
const unsigned short int * ip ;
int16_t offset ;
offset = - sys_tz . tz_minuteswest ;
if ( ! dest )
return NULL ;
2008-02-10 11:25:31 +01:00
dest - > typeAndTimezone = cpu_to_le16 ( 0x1000 | ( offset & 0x0FFF ) ) ;
2005-04-16 15:20:36 -07:00
ts . tv_sec + = offset * 60 ;
days = ts . tv_sec / SECS_PER_DAY ;
rem = ts . tv_sec % SECS_PER_DAY ;
dest - > hour = rem / SECS_PER_HOUR ;
rem % = SECS_PER_HOUR ;
dest - > minute = rem / 60 ;
dest - > second = rem % 60 ;
y = 1970 ;
2008-02-08 04:20:36 -08:00
# define DIV(a, b) ((a) / (b) - ((a) % (b) < 0))
2005-04-16 15:20:36 -07:00
# define LEAPS_THRU_END_OF(y) (DIV (y, 4) - DIV (y, 100) + DIV (y, 400))
2007-07-19 01:47:43 -07:00
while ( days < 0 | | days > = ( __isleap ( y ) ? 366 : 365 ) ) {
2005-04-16 15:20:36 -07:00
long int yg = y + days / 365 - ( days % 365 < 0 ) ;
/* Adjust DAYS and Y to match the guessed year. */
2007-07-21 04:37:18 -07:00
days - = ( ( yg - y ) * 365
2008-02-08 04:20:36 -08:00
+ LEAPS_THRU_END_OF ( yg - 1 )
- LEAPS_THRU_END_OF ( y - 1 ) ) ;
2005-04-16 15:20:36 -07:00
y = yg ;
}
2008-02-10 11:25:31 +01:00
dest - > year = cpu_to_le16 ( y ) ;
2005-04-16 15:20:36 -07:00
ip = __mon_yday [ __isleap ( y ) ] ;
2007-07-19 01:47:43 -07:00
for ( y = 11 ; days < ( long int ) ip [ y ] ; - - y )
2005-04-16 15:20:36 -07:00
continue ;
days - = ip [ y ] ;
dest - > month = y + 1 ;
dest - > day = days + 1 ;
dest - > centiseconds = ts . tv_nsec / 10000000 ;
2008-02-08 04:20:36 -08:00
dest - > hundredsOfMicroseconds = ( ts . tv_nsec / 1000 -
dest - > centiseconds * 10000 ) / 100 ;
2007-07-21 04:37:18 -07:00
dest - > microseconds = ( ts . tv_nsec / 1000 - dest - > centiseconds * 10000 -
dest - > hundredsOfMicroseconds * 100 ) ;
2005-04-16 15:20:36 -07:00
return dest ;
}
/* EOF */