summaryrefslogtreecommitdiff
path: root/gcc/ada/g-calend.adb
diff options
context:
space:
mode:
authorcharlet <charlet@138bc75d-0d04-0410-961f-82ee72b054a4>2007-09-26 10:43:45 +0000
committercharlet <charlet@138bc75d-0d04-0410-961f-82ee72b054a4>2007-09-26 10:43:45 +0000
commitad2eba2657654b8a011c66d3a06bdbdb43033516 (patch)
treee707496e9d85fb88cda71972a74e18f56f9a75e6 /gcc/ada/g-calend.adb
parentd8bd83391eb83e037875487d5b200375348c3994 (diff)
downloadgcc-ad2eba2657654b8a011c66d3a06bdbdb43033516.tar.gz
2007-09-26 Hristian Kirtchev <kirtchev@adacore.com>
* g-calend.adb (Has_53_Weeks): Rename to Last_Year_Has_53_Weeks. Add a call to Jan_1_Day _Of_Week to optimize its performance. (Is_Leap): Move the routine to the scope of Week_In_Year. (Jan_1_Day_Of_Week): New routine in Week_In_Year which calculates the weekday on which January 1 falls of Year - 1 and Year + 1. This function avoids calling Time_Of and Split, thus making it more efficent. (Week_In_Year): Reimplemented in oder to follow ISO 8601. * g-calend.ads (Week_In_Year): Change comment to reflect new implementation. git-svn-id: svn+ssh://gcc.gnu.org/svn/gcc/trunk@128790 138bc75d-0d04-0410-961f-82ee72b054a4
Diffstat (limited to 'gcc/ada/g-calend.adb')
-rw-r--r--gcc/ada/g-calend.adb248
1 files changed, 235 insertions, 13 deletions
diff --git a/gcc/ada/g-calend.adb b/gcc/ada/g-calend.adb
index 3fec4a093e8..f34a0d9d7c9 100644
--- a/gcc/ada/g-calend.adb
+++ b/gcc/ada/g-calend.adb
@@ -6,7 +6,7 @@
-- --
-- B o d y --
-- --
--- Copyright (C) 1999-2006, AdaCore --
+-- Copyright (C) 1999-2007, AdaCore --
-- --
-- GNAT is free software; you can redistribute it and/or modify it under --
-- terms of the GNU General Public License as published by the Free Soft- --
@@ -45,10 +45,8 @@ package body GNAT.Calendar is
Month : Month_Number;
Day : Day_Number;
Day_Secs : Day_Duration;
-
begin
Split (Date, Year, Month, Day, Day_Secs);
-
return Julian_Day (Year, Month, Day) - Julian_Day (Year, 1, 1) + 1;
end Day_In_Year;
@@ -61,10 +59,8 @@ package body GNAT.Calendar is
Month : Month_Number;
Day : Day_Number;
Day_Secs : Day_Duration;
-
begin
Split (Date, Year, Month, Day, Day_Secs);
-
return Day_Name'Val ((Julian_Day (Year, Month, Day)) mod 7);
end Day_Of_Week;
@@ -80,7 +76,6 @@ package body GNAT.Calendar is
Minute : Minute_Number;
Second : Second_Number;
Sub_Second : Second_Duration;
-
begin
Split (Date, Year, Month, Day, Hour, Minute, Second, Sub_Second);
return Hour;
@@ -140,7 +135,6 @@ package body GNAT.Calendar is
Minute : Minute_Number;
Second : Second_Number;
Sub_Second : Second_Duration;
-
begin
Split (Date, Year, Month, Day, Hour, Minute, Second, Sub_Second);
return Minute;
@@ -158,7 +152,6 @@ package body GNAT.Calendar is
Minute : Minute_Number;
Second : Second_Number;
Sub_Second : Second_Duration;
-
begin
Split (Date, Year, Month, Day, Hour, Minute, Second, Sub_Second);
return Second;
@@ -209,7 +202,6 @@ package body GNAT.Calendar is
Minute : Minute_Number;
Second : Second_Number;
Sub_Second : Second_Duration;
-
begin
Split (Date, Year, Month, Day, Hour, Minute, Second, Sub_Second);
return Sub_Second;
@@ -301,16 +293,246 @@ package body GNAT.Calendar is
Minute : Minute_Number;
Second : Second_Number;
Sub_Second : Second_Duration;
- Offset : Natural;
+ Jan_1 : Day_Name;
+ Shift : Week_In_Year_Number;
+ Start_Week : Week_In_Year_Number;
+
+ function Is_Leap (Year : Year_Number) return Boolean;
+ -- Return True if Year denotes a leap year. Leap centential years are
+ -- properly handled.
+
+ function Jan_1_Day_Of_Week
+ (Jan_1 : Day_Name;
+ Year : Year_Number;
+ Last_Year : Boolean := False;
+ Next_Year : Boolean := False) return Day_Name;
+ -- Given the weekday of January 1 in Year, determine the weekday on
+ -- which January 1 fell last year or will fall next year as set by
+ -- the two flags. This routine does not call Time_Of or Split.
+
+ function Last_Year_Has_53_Weeks
+ (Jan_1 : Day_Name;
+ Year : Year_Number) return Boolean;
+ -- Given the weekday of January 1 in Year, determine whether last year
+ -- has 53 weeks. A False value implies that the year has 52 weeks.
+
+ -------------
+ -- Is_Leap --
+ -------------
+
+ function Is_Leap (Year : Year_Number) return Boolean is
+ begin
+ if Year mod 400 = 0 then
+ return True;
+ elsif Year mod 100 = 0 then
+ return False;
+ else
+ return Year mod 4 = 0;
+ end if;
+ end Is_Leap;
+
+ -----------------------
+ -- Jan_1_Day_Of_Week --
+ -----------------------
+
+ function Jan_1_Day_Of_Week
+ (Jan_1 : Day_Name;
+ Year : Year_Number;
+ Last_Year : Boolean := False;
+ Next_Year : Boolean := False) return Day_Name
+ is
+ Shift : Integer := 0;
+
+ begin
+ if Last_Year then
+ if Is_Leap (Year - 1) then
+ Shift := -2;
+ else
+ Shift := -1;
+ end if;
+
+ elsif Next_Year then
+ if Is_Leap (Year) then
+ Shift := 2;
+ else
+ Shift := 1;
+ end if;
+ end if;
+
+ return Day_Name'Val ((Day_Name'Pos (Jan_1) + Shift) mod 7);
+ end Jan_1_Day_Of_Week;
+
+ ----------------------------
+ -- Last_Year_Has_53_Weeks --
+ ----------------------------
+
+ function Last_Year_Has_53_Weeks
+ (Jan_1 : Day_Name;
+ Year : Year_Number) return Boolean
+ is
+ Last_Jan_1 : constant Day_Name :=
+ Jan_1_Day_Of_Week (Jan_1, Year, Last_Year => True);
+ begin
+ -- These two cases are illustrated in the table below
+
+ return
+ Last_Jan_1 = Thursday
+ or else
+ (Last_Jan_1 = Wednesday
+ and then Is_Leap (Year - 1));
+ end Last_Year_Has_53_Weeks;
+
+ -- Start of processing for Week_In_Year
begin
Split (Date, Year, Month, Day, Hour, Minute, Second, Sub_Second);
- -- Day offset number for the first week of the year
+ -- According to ISO 8601, the first week of year Y is the week that
+ -- contains the first Thursday in year Y. The following table contains
+ -- all possible combinations of years and weekdays along with examples.
+
+ -- +-------+------+-------+---------+
+ -- | Jan 1 | Leap | Weeks | Example |
+ -- +-------+------+-------+---------+
+ -- | Mon | No | 52 | 2007 |
+ -- +-------+------+-------+---------+
+ -- | Mon | Yes | 52 | 1996 |
+ -- +-------+------+-------+---------+
+ -- | Tue | No | 52 | 2002 |
+ -- +-------+------+-------+---------+
+ -- | Tue | Yes | 52 | 1980 |
+ -- +-------+------+-------+---------+
+ -- | Wed | No | 52 | 2003 |
+ -- +-------+------#########---------+
+ -- | Wed | Yes # 53 # 1992 |
+ -- +-------+------#-------#---------+
+ -- | Thu | No # 53 # 1998 |
+ -- +-------+------#-------#---------+
+ -- | Thu | Yes # 53 # 2004 |
+ -- +-------+------#########---------+
+ -- | Fri | No | 52 | 1999 |
+ -- +-------+------+-------+---------+
+ -- | Fri | Yes | 52 | 1988 |
+ -- +-------+------+-------+---------+
+ -- | Sat | No | 52 | 1994 |
+ -- +-------+------+-------+---------+
+ -- | Sat | Yes | 52 | 1972 |
+ -- +-------+------+-------+---------+
+ -- | Sun | No | 52 | 1995 |
+ -- +-------+------+-------+---------+
+ -- | Sun | Yes | 52 | 1956 |
+ -- +-------+------+-------+---------+
+
+ -- A small optimization, the input date is January 1. Note that this
+ -- is a key day since it determines the number of weeks and is used
+ -- when special casing the first week of January and the last week of
+ -- December.
+
+ if Day = 1
+ and then Month = 1
+ then
+ Jan_1 := Day_Of_Week (Date);
+ else
+ Jan_1 := Day_Of_Week (Time_Of (Year, 1, 1, 0.0));
+ end if;
+
+ if Month = 1 then
+
+ -- Special case 1: January 1, 2 and 3. These three days may belong
+ -- to last year's last week which can be week number 52 or 53.
+
+ -- +-----+-----+-----+=====+-----+-----+-----+
+ -- | Mon | Tue | Wed # Thu # Fri | Sat | Sun |
+ -- +-----+-----+-----+-----+-----+-----+-----+
+ -- | 26 | 27 | 28 # 29 # 30 | 31 | 1 |
+ -- +-----+-----+-----+-----+-----+-----+-----+
+ -- | 27 | 28 | 29 # 30 # 31 | 1 | 2 |
+ -- +-----+-----+-----+-----+-----+-----+-----+
+ -- | 28 | 29 | 30 # 31 # 1 | 2 | 3 |
+ -- +-----+-----+-----+=====+-----+-----+-----+
+
+ if (Day = 1 and then Jan_1 in Friday .. Sunday)
+ or else
+ (Day = 2 and then Jan_1 in Friday .. Saturday)
+ or else
+ (Day = 3 and then Jan_1 = Friday)
+ then
+ if Last_Year_Has_53_Weeks (Jan_1, Year) then
+ return 53;
+ else
+ return 52;
+ end if;
+
+ -- Special case 2: January 1, 2, 3, 4, 5 and 6 of the first week. In
+ -- this scenario January 1 does not fall on a Monday.
+
+ -- +-----+-----+-----+=====+-----+-----+-----+
+ -- | Mon | Tue | Wed # Thu # Fri | Sat | Sun |
+ -- +-----+-----+-----+-----+-----+-----+-----+
+ -- | 29 | 30 | 31 # 1 # 2 | 3 | 4 |
+ -- +-----+-----+-----+-----+-----+-----+-----+
+ -- | 30 | 31 | 1 # 2 # 3 | 4 | 5 |
+ -- +-----+-----+-----+-----+-----+-----+-----+
+ -- | 31 | 1 | 2 # 3 # 4 | 5 | 6 |
+ -- +-----+-----+-----+-----+-----+-----+-----+
+
+ elsif (Day <= 4 and then Jan_1 in Tuesday .. Thursday)
+ or else
+ (Day = 5 and then Jan_1 in Tuesday .. Wednesday)
+ or else
+ (Day = 6 and then Jan_1 = Tuesday)
+ then
+ return 1;
+ end if;
+
+ -- Special case 3: December 29, 30 and 31. These days may belong to
+ -- next year's first week.
+
+ -- +-----+-----+-----+=====+-----+-----+-----+
+ -- | Mon | Tue | Wed # Thu # Fri | Sat | Sun |
+ -- +-----+-----+-----+-----+-----+-----+-----+
+ -- | 29 | 30 | 31 # 1 # 2 | 3 | 4 |
+ -- +-----+-----+-----+-----+-----+-----+-----+
+ -- | 30 | 31 | 1 # 2 # 3 | 4 | 5 |
+ -- +-----+-----+-----+-----+-----+-----+-----+
+ -- | 31 | 1 | 2 # 3 # 4 | 5 | 6 |
+ -- +-----+-----+-----+=====+-----+-----+-----+
+
+ elsif Month = 12
+ and then Day > 28
+ then
+ declare
+ Next_Jan_1 : constant Day_Name :=
+ Jan_1_Day_Of_Week (Jan_1, Year, Next_Year => True);
+ begin
+ if (Day = 29 and then Next_Jan_1 = Thursday)
+ or else
+ (Day = 30 and then Next_Jan_1 in Wednesday .. Thursday)
+ or else
+ (Day = 31 and then Next_Jan_1 in Tuesday .. Thursday)
+ then
+ return 1;
+ end if;
+ end;
+ end if;
+
+ -- Determine the week from which to start counting. If January 1 does
+ -- not belong to the first week of the input year, then the next week
+ -- is the first week.
+
+ if Jan_1 in Friday .. Sunday then
+ Start_Week := 1;
+ else
+ Start_Week := 2;
+ end if;
- Offset := Julian_Day (Year, 1, 1) mod 7;
+ -- At this point all special combinations have been accounted for and
+ -- the proper start week has been found. Since January 1 may not fall
+ -- on a Monday, shift 7 - Day_Name'Pos (Jan_1). This action ensures an
+ -- origin which falls on Monday.
- return 1 + ((Day_In_Year (Date) - 1) + Offset) / 7;
+ Shift := 7 - Day_Name'Pos (Jan_1);
+ return Start_Week + (Day_In_Year (Date) - Shift - 1) / 7;
end Week_In_Year;
end GNAT.Calendar;