annotate gcc/ada/libgnat/g-calend.adb @ 131:84e7813d76e9

gcc-8.2
author mir3636
date Thu, 25 Oct 2018 07:37:49 +0900
parents 04ced10e8804
children 1830386684a0
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
111
kono
parents:
diff changeset
1 ------------------------------------------------------------------------------
kono
parents:
diff changeset
2 -- --
kono
parents:
diff changeset
3 -- GNAT RUN-TIME COMPONENTS --
kono
parents:
diff changeset
4 -- --
kono
parents:
diff changeset
5 -- G N A T . C A L E N D A R --
kono
parents:
diff changeset
6 -- --
kono
parents:
diff changeset
7 -- B o d y --
kono
parents:
diff changeset
8 -- --
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
9 -- Copyright (C) 1999-2018, AdaCore --
111
kono
parents:
diff changeset
10 -- --
kono
parents:
diff changeset
11 -- GNAT is free software; you can redistribute it and/or modify it under --
kono
parents:
diff changeset
12 -- terms of the GNU General Public License as published by the Free Soft- --
kono
parents:
diff changeset
13 -- ware Foundation; either version 3, or (at your option) any later ver- --
kono
parents:
diff changeset
14 -- sion. GNAT is distributed in the hope that it will be useful, but WITH- --
kono
parents:
diff changeset
15 -- OUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY --
kono
parents:
diff changeset
16 -- or FITNESS FOR A PARTICULAR PURPOSE. --
kono
parents:
diff changeset
17 -- --
kono
parents:
diff changeset
18 -- As a special exception under Section 7 of GPL version 3, you are granted --
kono
parents:
diff changeset
19 -- additional permissions described in the GCC Runtime Library Exception, --
kono
parents:
diff changeset
20 -- version 3.1, as published by the Free Software Foundation. --
kono
parents:
diff changeset
21 -- --
kono
parents:
diff changeset
22 -- You should have received a copy of the GNU General Public License and --
kono
parents:
diff changeset
23 -- a copy of the GCC Runtime Library Exception along with this program; --
kono
parents:
diff changeset
24 -- see the files COPYING3 and COPYING.RUNTIME respectively. If not, see --
kono
parents:
diff changeset
25 -- <http://www.gnu.org/licenses/>. --
kono
parents:
diff changeset
26 -- --
kono
parents:
diff changeset
27 -- GNAT was originally developed by the GNAT team at New York University. --
kono
parents:
diff changeset
28 -- Extensive contributions were provided by Ada Core Technologies Inc. --
kono
parents:
diff changeset
29 -- --
kono
parents:
diff changeset
30 ------------------------------------------------------------------------------
kono
parents:
diff changeset
31
kono
parents:
diff changeset
32 with Interfaces.C.Extensions;
kono
parents:
diff changeset
33
kono
parents:
diff changeset
34 package body GNAT.Calendar is
kono
parents:
diff changeset
35 use Ada.Calendar;
kono
parents:
diff changeset
36 use Interfaces;
kono
parents:
diff changeset
37
kono
parents:
diff changeset
38 -----------------
kono
parents:
diff changeset
39 -- Day_In_Year --
kono
parents:
diff changeset
40 -----------------
kono
parents:
diff changeset
41
kono
parents:
diff changeset
42 function Day_In_Year (Date : Time) return Day_In_Year_Number is
kono
parents:
diff changeset
43 Year : Year_Number;
kono
parents:
diff changeset
44 Month : Month_Number;
kono
parents:
diff changeset
45 Day : Day_Number;
kono
parents:
diff changeset
46 Day_Secs : Day_Duration;
kono
parents:
diff changeset
47 pragma Unreferenced (Day_Secs);
kono
parents:
diff changeset
48 begin
kono
parents:
diff changeset
49 Split (Date, Year, Month, Day, Day_Secs);
kono
parents:
diff changeset
50 return Julian_Day (Year, Month, Day) - Julian_Day (Year, 1, 1) + 1;
kono
parents:
diff changeset
51 end Day_In_Year;
kono
parents:
diff changeset
52
kono
parents:
diff changeset
53 -----------------
kono
parents:
diff changeset
54 -- Day_Of_Week --
kono
parents:
diff changeset
55 -----------------
kono
parents:
diff changeset
56
kono
parents:
diff changeset
57 function Day_Of_Week (Date : Time) return Day_Name is
kono
parents:
diff changeset
58 Year : Year_Number;
kono
parents:
diff changeset
59 Month : Month_Number;
kono
parents:
diff changeset
60 Day : Day_Number;
kono
parents:
diff changeset
61 Day_Secs : Day_Duration;
kono
parents:
diff changeset
62 pragma Unreferenced (Day_Secs);
kono
parents:
diff changeset
63 begin
kono
parents:
diff changeset
64 Split (Date, Year, Month, Day, Day_Secs);
kono
parents:
diff changeset
65 return Day_Name'Val ((Julian_Day (Year, Month, Day)) mod 7);
kono
parents:
diff changeset
66 end Day_Of_Week;
kono
parents:
diff changeset
67
kono
parents:
diff changeset
68 ----------
kono
parents:
diff changeset
69 -- Hour --
kono
parents:
diff changeset
70 ----------
kono
parents:
diff changeset
71
kono
parents:
diff changeset
72 function Hour (Date : Time) return Hour_Number is
kono
parents:
diff changeset
73 Year : Year_Number;
kono
parents:
diff changeset
74 Month : Month_Number;
kono
parents:
diff changeset
75 Day : Day_Number;
kono
parents:
diff changeset
76 Hour : Hour_Number;
kono
parents:
diff changeset
77 Minute : Minute_Number;
kono
parents:
diff changeset
78 Second : Second_Number;
kono
parents:
diff changeset
79 Sub_Second : Second_Duration;
kono
parents:
diff changeset
80 pragma Unreferenced (Year, Month, Day, Minute, Second, Sub_Second);
kono
parents:
diff changeset
81 begin
kono
parents:
diff changeset
82 Split (Date, Year, Month, Day, Hour, Minute, Second, Sub_Second);
kono
parents:
diff changeset
83 return Hour;
kono
parents:
diff changeset
84 end Hour;
kono
parents:
diff changeset
85
kono
parents:
diff changeset
86 ----------------
kono
parents:
diff changeset
87 -- Julian_Day --
kono
parents:
diff changeset
88 ----------------
kono
parents:
diff changeset
89
kono
parents:
diff changeset
90 -- Julian_Day is used to by Day_Of_Week and Day_In_Year. Note that this
kono
parents:
diff changeset
91 -- implementation is not expensive.
kono
parents:
diff changeset
92
kono
parents:
diff changeset
93 function Julian_Day
kono
parents:
diff changeset
94 (Year : Year_Number;
kono
parents:
diff changeset
95 Month : Month_Number;
kono
parents:
diff changeset
96 Day : Day_Number) return Integer
kono
parents:
diff changeset
97 is
kono
parents:
diff changeset
98 Internal_Year : Integer;
kono
parents:
diff changeset
99 Internal_Month : Integer;
kono
parents:
diff changeset
100 Internal_Day : Integer;
kono
parents:
diff changeset
101 Julian_Date : Integer;
kono
parents:
diff changeset
102 C : Integer;
kono
parents:
diff changeset
103 Ya : Integer;
kono
parents:
diff changeset
104
kono
parents:
diff changeset
105 begin
kono
parents:
diff changeset
106 Internal_Year := Integer (Year);
kono
parents:
diff changeset
107 Internal_Month := Integer (Month);
kono
parents:
diff changeset
108 Internal_Day := Integer (Day);
kono
parents:
diff changeset
109
kono
parents:
diff changeset
110 if Internal_Month > 2 then
kono
parents:
diff changeset
111 Internal_Month := Internal_Month - 3;
kono
parents:
diff changeset
112 else
kono
parents:
diff changeset
113 Internal_Month := Internal_Month + 9;
kono
parents:
diff changeset
114 Internal_Year := Internal_Year - 1;
kono
parents:
diff changeset
115 end if;
kono
parents:
diff changeset
116
kono
parents:
diff changeset
117 C := Internal_Year / 100;
kono
parents:
diff changeset
118 Ya := Internal_Year - (100 * C);
kono
parents:
diff changeset
119
kono
parents:
diff changeset
120 Julian_Date := (146_097 * C) / 4 +
kono
parents:
diff changeset
121 (1_461 * Ya) / 4 +
kono
parents:
diff changeset
122 (153 * Internal_Month + 2) / 5 +
kono
parents:
diff changeset
123 Internal_Day + 1_721_119;
kono
parents:
diff changeset
124
kono
parents:
diff changeset
125 return Julian_Date;
kono
parents:
diff changeset
126 end Julian_Day;
kono
parents:
diff changeset
127
kono
parents:
diff changeset
128 ------------
kono
parents:
diff changeset
129 -- Minute --
kono
parents:
diff changeset
130 ------------
kono
parents:
diff changeset
131
kono
parents:
diff changeset
132 function Minute (Date : Time) return Minute_Number is
kono
parents:
diff changeset
133 Year : Year_Number;
kono
parents:
diff changeset
134 Month : Month_Number;
kono
parents:
diff changeset
135 Day : Day_Number;
kono
parents:
diff changeset
136 Hour : Hour_Number;
kono
parents:
diff changeset
137 Minute : Minute_Number;
kono
parents:
diff changeset
138 Second : Second_Number;
kono
parents:
diff changeset
139 Sub_Second : Second_Duration;
kono
parents:
diff changeset
140 pragma Unreferenced (Year, Month, Day, Hour, Second, Sub_Second);
kono
parents:
diff changeset
141 begin
kono
parents:
diff changeset
142 Split (Date, Year, Month, Day, Hour, Minute, Second, Sub_Second);
kono
parents:
diff changeset
143 return Minute;
kono
parents:
diff changeset
144 end Minute;
kono
parents:
diff changeset
145
kono
parents:
diff changeset
146 ------------
kono
parents:
diff changeset
147 -- Second --
kono
parents:
diff changeset
148 ------------
kono
parents:
diff changeset
149
kono
parents:
diff changeset
150 function Second (Date : Time) return Second_Number is
kono
parents:
diff changeset
151 Year : Year_Number;
kono
parents:
diff changeset
152 Month : Month_Number;
kono
parents:
diff changeset
153 Day : Day_Number;
kono
parents:
diff changeset
154 Hour : Hour_Number;
kono
parents:
diff changeset
155 Minute : Minute_Number;
kono
parents:
diff changeset
156 Second : Second_Number;
kono
parents:
diff changeset
157 Sub_Second : Second_Duration;
kono
parents:
diff changeset
158 pragma Unreferenced (Year, Month, Day, Hour, Minute, Sub_Second);
kono
parents:
diff changeset
159 begin
kono
parents:
diff changeset
160 Split (Date, Year, Month, Day, Hour, Minute, Second, Sub_Second);
kono
parents:
diff changeset
161 return Second;
kono
parents:
diff changeset
162 end Second;
kono
parents:
diff changeset
163
kono
parents:
diff changeset
164 -----------
kono
parents:
diff changeset
165 -- Split --
kono
parents:
diff changeset
166 -----------
kono
parents:
diff changeset
167
kono
parents:
diff changeset
168 procedure Split
kono
parents:
diff changeset
169 (Date : Time;
kono
parents:
diff changeset
170 Year : out Year_Number;
kono
parents:
diff changeset
171 Month : out Month_Number;
kono
parents:
diff changeset
172 Day : out Day_Number;
kono
parents:
diff changeset
173 Hour : out Hour_Number;
kono
parents:
diff changeset
174 Minute : out Minute_Number;
kono
parents:
diff changeset
175 Second : out Second_Number;
kono
parents:
diff changeset
176 Sub_Second : out Second_Duration)
kono
parents:
diff changeset
177 is
kono
parents:
diff changeset
178 Day_Secs : Day_Duration;
kono
parents:
diff changeset
179 Secs : Natural;
kono
parents:
diff changeset
180
kono
parents:
diff changeset
181 begin
kono
parents:
diff changeset
182 Split (Date, Year, Month, Day, Day_Secs);
kono
parents:
diff changeset
183
kono
parents:
diff changeset
184 Secs := (if Day_Secs = 0.0 then 0 else Natural (Day_Secs - 0.5));
kono
parents:
diff changeset
185 Sub_Second := Second_Duration (Day_Secs - Day_Duration (Secs));
kono
parents:
diff changeset
186 Hour := Hour_Number (Secs / 3_600);
kono
parents:
diff changeset
187 Secs := Secs mod 3_600;
kono
parents:
diff changeset
188 Minute := Minute_Number (Secs / 60);
kono
parents:
diff changeset
189 Second := Second_Number (Secs mod 60);
kono
parents:
diff changeset
190 end Split;
kono
parents:
diff changeset
191
kono
parents:
diff changeset
192 ---------------------
kono
parents:
diff changeset
193 -- Split_At_Locale --
kono
parents:
diff changeset
194 ---------------------
kono
parents:
diff changeset
195
kono
parents:
diff changeset
196 procedure Split_At_Locale
kono
parents:
diff changeset
197 (Date : Time;
kono
parents:
diff changeset
198 Year : out Year_Number;
kono
parents:
diff changeset
199 Month : out Month_Number;
kono
parents:
diff changeset
200 Day : out Day_Number;
kono
parents:
diff changeset
201 Hour : out Hour_Number;
kono
parents:
diff changeset
202 Minute : out Minute_Number;
kono
parents:
diff changeset
203 Second : out Second_Number;
kono
parents:
diff changeset
204 Sub_Second : out Second_Duration)
kono
parents:
diff changeset
205 is
kono
parents:
diff changeset
206 procedure Ada_Calendar_Split
kono
parents:
diff changeset
207 (Date : Time;
kono
parents:
diff changeset
208 Year : out Year_Number;
kono
parents:
diff changeset
209 Month : out Month_Number;
kono
parents:
diff changeset
210 Day : out Day_Number;
kono
parents:
diff changeset
211 Day_Secs : out Day_Duration;
kono
parents:
diff changeset
212 Hour : out Integer;
kono
parents:
diff changeset
213 Minute : out Integer;
kono
parents:
diff changeset
214 Second : out Integer;
kono
parents:
diff changeset
215 Sub_Sec : out Duration;
kono
parents:
diff changeset
216 Leap_Sec : out Boolean;
kono
parents:
diff changeset
217 Use_TZ : Boolean;
kono
parents:
diff changeset
218 Is_Historic : Boolean;
kono
parents:
diff changeset
219 Time_Zone : Long_Integer);
kono
parents:
diff changeset
220 pragma Import (Ada, Ada_Calendar_Split, "__gnat_split");
kono
parents:
diff changeset
221
kono
parents:
diff changeset
222 Ds : Day_Duration;
kono
parents:
diff changeset
223 Le : Boolean;
kono
parents:
diff changeset
224
kono
parents:
diff changeset
225 pragma Unreferenced (Ds, Le);
kono
parents:
diff changeset
226
kono
parents:
diff changeset
227 begin
kono
parents:
diff changeset
228 -- Even though the input time zone is UTC (0), the flag Use_TZ will
kono
parents:
diff changeset
229 -- ensure that Split picks up the local time zone.
kono
parents:
diff changeset
230
kono
parents:
diff changeset
231 Ada_Calendar_Split
kono
parents:
diff changeset
232 (Date => Date,
kono
parents:
diff changeset
233 Year => Year,
kono
parents:
diff changeset
234 Month => Month,
kono
parents:
diff changeset
235 Day => Day,
kono
parents:
diff changeset
236 Day_Secs => Ds,
kono
parents:
diff changeset
237 Hour => Hour,
kono
parents:
diff changeset
238 Minute => Minute,
kono
parents:
diff changeset
239 Second => Second,
kono
parents:
diff changeset
240 Sub_Sec => Sub_Second,
kono
parents:
diff changeset
241 Leap_Sec => Le,
kono
parents:
diff changeset
242 Use_TZ => False,
kono
parents:
diff changeset
243 Is_Historic => False,
kono
parents:
diff changeset
244 Time_Zone => 0);
kono
parents:
diff changeset
245 end Split_At_Locale;
kono
parents:
diff changeset
246
kono
parents:
diff changeset
247 ----------------
kono
parents:
diff changeset
248 -- Sub_Second --
kono
parents:
diff changeset
249 ----------------
kono
parents:
diff changeset
250
kono
parents:
diff changeset
251 function Sub_Second (Date : Time) return Second_Duration is
kono
parents:
diff changeset
252 Year : Year_Number;
kono
parents:
diff changeset
253 Month : Month_Number;
kono
parents:
diff changeset
254 Day : Day_Number;
kono
parents:
diff changeset
255 Hour : Hour_Number;
kono
parents:
diff changeset
256 Minute : Minute_Number;
kono
parents:
diff changeset
257 Second : Second_Number;
kono
parents:
diff changeset
258 Sub_Second : Second_Duration;
kono
parents:
diff changeset
259 pragma Unreferenced (Year, Month, Day, Hour, Minute, Second);
kono
parents:
diff changeset
260 begin
kono
parents:
diff changeset
261 Split (Date, Year, Month, Day, Hour, Minute, Second, Sub_Second);
kono
parents:
diff changeset
262 return Sub_Second;
kono
parents:
diff changeset
263 end Sub_Second;
kono
parents:
diff changeset
264
kono
parents:
diff changeset
265 -------------
kono
parents:
diff changeset
266 -- Time_Of --
kono
parents:
diff changeset
267 -------------
kono
parents:
diff changeset
268
kono
parents:
diff changeset
269 function Time_Of
kono
parents:
diff changeset
270 (Year : Year_Number;
kono
parents:
diff changeset
271 Month : Month_Number;
kono
parents:
diff changeset
272 Day : Day_Number;
kono
parents:
diff changeset
273 Hour : Hour_Number;
kono
parents:
diff changeset
274 Minute : Minute_Number;
kono
parents:
diff changeset
275 Second : Second_Number;
kono
parents:
diff changeset
276 Sub_Second : Second_Duration := 0.0) return Time
kono
parents:
diff changeset
277 is
kono
parents:
diff changeset
278 Day_Secs : constant Day_Duration :=
kono
parents:
diff changeset
279 Day_Duration (Hour * 3_600) +
kono
parents:
diff changeset
280 Day_Duration (Minute * 60) +
kono
parents:
diff changeset
281 Day_Duration (Second) +
kono
parents:
diff changeset
282 Sub_Second;
kono
parents:
diff changeset
283 begin
kono
parents:
diff changeset
284 return Time_Of (Year, Month, Day, Day_Secs);
kono
parents:
diff changeset
285 end Time_Of;
kono
parents:
diff changeset
286
kono
parents:
diff changeset
287 -----------------------
kono
parents:
diff changeset
288 -- Time_Of_At_Locale --
kono
parents:
diff changeset
289 -----------------------
kono
parents:
diff changeset
290
kono
parents:
diff changeset
291 function Time_Of_At_Locale
kono
parents:
diff changeset
292 (Year : Year_Number;
kono
parents:
diff changeset
293 Month : Month_Number;
kono
parents:
diff changeset
294 Day : Day_Number;
kono
parents:
diff changeset
295 Hour : Hour_Number;
kono
parents:
diff changeset
296 Minute : Minute_Number;
kono
parents:
diff changeset
297 Second : Second_Number;
kono
parents:
diff changeset
298 Sub_Second : Second_Duration := 0.0) return Time
kono
parents:
diff changeset
299 is
kono
parents:
diff changeset
300 function Ada_Calendar_Time_Of
kono
parents:
diff changeset
301 (Year : Year_Number;
kono
parents:
diff changeset
302 Month : Month_Number;
kono
parents:
diff changeset
303 Day : Day_Number;
kono
parents:
diff changeset
304 Day_Secs : Day_Duration;
kono
parents:
diff changeset
305 Hour : Integer;
kono
parents:
diff changeset
306 Minute : Integer;
kono
parents:
diff changeset
307 Second : Integer;
kono
parents:
diff changeset
308 Sub_Sec : Duration;
kono
parents:
diff changeset
309 Leap_Sec : Boolean;
kono
parents:
diff changeset
310 Use_Day_Secs : Boolean;
kono
parents:
diff changeset
311 Use_TZ : Boolean;
kono
parents:
diff changeset
312 Is_Historic : Boolean;
kono
parents:
diff changeset
313 Time_Zone : Long_Integer) return Time;
kono
parents:
diff changeset
314 pragma Import (Ada, Ada_Calendar_Time_Of, "__gnat_time_of");
kono
parents:
diff changeset
315
kono
parents:
diff changeset
316 begin
kono
parents:
diff changeset
317 -- Even though the input time zone is UTC (0), the flag Use_TZ will
kono
parents:
diff changeset
318 -- ensure that Split picks up the local time zone.
kono
parents:
diff changeset
319
kono
parents:
diff changeset
320 return
kono
parents:
diff changeset
321 Ada_Calendar_Time_Of
kono
parents:
diff changeset
322 (Year => Year,
kono
parents:
diff changeset
323 Month => Month,
kono
parents:
diff changeset
324 Day => Day,
kono
parents:
diff changeset
325 Day_Secs => 0.0,
kono
parents:
diff changeset
326 Hour => Hour,
kono
parents:
diff changeset
327 Minute => Minute,
kono
parents:
diff changeset
328 Second => Second,
kono
parents:
diff changeset
329 Sub_Sec => Sub_Second,
kono
parents:
diff changeset
330 Leap_Sec => False,
kono
parents:
diff changeset
331 Use_Day_Secs => False,
kono
parents:
diff changeset
332 Use_TZ => False,
kono
parents:
diff changeset
333 Is_Historic => False,
kono
parents:
diff changeset
334 Time_Zone => 0);
kono
parents:
diff changeset
335 end Time_Of_At_Locale;
kono
parents:
diff changeset
336
kono
parents:
diff changeset
337 -----------------
kono
parents:
diff changeset
338 -- To_Duration --
kono
parents:
diff changeset
339 -----------------
kono
parents:
diff changeset
340
kono
parents:
diff changeset
341 function To_Duration (T : not null access timeval) return Duration is
kono
parents:
diff changeset
342
kono
parents:
diff changeset
343 procedure timeval_to_duration
kono
parents:
diff changeset
344 (T : not null access timeval;
kono
parents:
diff changeset
345 sec : not null access C.Extensions.long_long;
kono
parents:
diff changeset
346 usec : not null access C.long);
kono
parents:
diff changeset
347 pragma Import (C, timeval_to_duration, "__gnat_timeval_to_duration");
kono
parents:
diff changeset
348
kono
parents:
diff changeset
349 Micro : constant := 10**6;
kono
parents:
diff changeset
350 sec : aliased C.Extensions.long_long;
kono
parents:
diff changeset
351 usec : aliased C.long;
kono
parents:
diff changeset
352
kono
parents:
diff changeset
353 begin
kono
parents:
diff changeset
354 timeval_to_duration (T, sec'Access, usec'Access);
kono
parents:
diff changeset
355 return Duration (sec) + Duration (usec) / Micro;
kono
parents:
diff changeset
356 end To_Duration;
kono
parents:
diff changeset
357
kono
parents:
diff changeset
358 ----------------
kono
parents:
diff changeset
359 -- To_Timeval --
kono
parents:
diff changeset
360 ----------------
kono
parents:
diff changeset
361
kono
parents:
diff changeset
362 function To_Timeval (D : Duration) return timeval is
kono
parents:
diff changeset
363
kono
parents:
diff changeset
364 procedure duration_to_timeval
kono
parents:
diff changeset
365 (Sec : C.Extensions.long_long;
kono
parents:
diff changeset
366 Usec : C.long;
kono
parents:
diff changeset
367 T : not null access timeval);
kono
parents:
diff changeset
368 pragma Import (C, duration_to_timeval, "__gnat_duration_to_timeval");
kono
parents:
diff changeset
369
kono
parents:
diff changeset
370 Micro : constant := 10**6;
kono
parents:
diff changeset
371 Result : aliased timeval;
kono
parents:
diff changeset
372 sec : C.Extensions.long_long;
kono
parents:
diff changeset
373 usec : C.long;
kono
parents:
diff changeset
374
kono
parents:
diff changeset
375 begin
kono
parents:
diff changeset
376 if D = 0.0 then
kono
parents:
diff changeset
377 sec := 0;
kono
parents:
diff changeset
378 usec := 0;
kono
parents:
diff changeset
379 else
kono
parents:
diff changeset
380 sec := C.Extensions.long_long (D - 0.5);
kono
parents:
diff changeset
381 usec := C.long ((D - Duration (sec)) * Micro - 0.5);
kono
parents:
diff changeset
382 end if;
kono
parents:
diff changeset
383
kono
parents:
diff changeset
384 duration_to_timeval (sec, usec, Result'Access);
kono
parents:
diff changeset
385
kono
parents:
diff changeset
386 return Result;
kono
parents:
diff changeset
387 end To_Timeval;
kono
parents:
diff changeset
388
kono
parents:
diff changeset
389 ------------------
kono
parents:
diff changeset
390 -- Week_In_Year --
kono
parents:
diff changeset
391 ------------------
kono
parents:
diff changeset
392
kono
parents:
diff changeset
393 function Week_In_Year (Date : Time) return Week_In_Year_Number is
kono
parents:
diff changeset
394 Year : Year_Number;
kono
parents:
diff changeset
395 Week : Week_In_Year_Number;
kono
parents:
diff changeset
396 pragma Unreferenced (Year);
kono
parents:
diff changeset
397 begin
kono
parents:
diff changeset
398 Year_Week_In_Year (Date, Year, Week);
kono
parents:
diff changeset
399 return Week;
kono
parents:
diff changeset
400 end Week_In_Year;
kono
parents:
diff changeset
401
kono
parents:
diff changeset
402 -----------------------
kono
parents:
diff changeset
403 -- Year_Week_In_Year --
kono
parents:
diff changeset
404 -----------------------
kono
parents:
diff changeset
405
kono
parents:
diff changeset
406 procedure Year_Week_In_Year
kono
parents:
diff changeset
407 (Date : Time;
kono
parents:
diff changeset
408 Year : out Year_Number;
kono
parents:
diff changeset
409 Week : out Week_In_Year_Number)
kono
parents:
diff changeset
410 is
kono
parents:
diff changeset
411 Month : Month_Number;
kono
parents:
diff changeset
412 Day : Day_Number;
kono
parents:
diff changeset
413 Hour : Hour_Number;
kono
parents:
diff changeset
414 Minute : Minute_Number;
kono
parents:
diff changeset
415 Second : Second_Number;
kono
parents:
diff changeset
416 Sub_Second : Second_Duration;
kono
parents:
diff changeset
417 Jan_1 : Day_Name;
kono
parents:
diff changeset
418 Shift : Week_In_Year_Number;
kono
parents:
diff changeset
419 Start_Week : Week_In_Year_Number;
kono
parents:
diff changeset
420
kono
parents:
diff changeset
421 pragma Unreferenced (Hour, Minute, Second, Sub_Second);
kono
parents:
diff changeset
422
kono
parents:
diff changeset
423 function Is_Leap (Year : Year_Number) return Boolean;
kono
parents:
diff changeset
424 -- Return True if Year denotes a leap year. Leap centennial years are
kono
parents:
diff changeset
425 -- properly handled.
kono
parents:
diff changeset
426
kono
parents:
diff changeset
427 function Jan_1_Day_Of_Week
kono
parents:
diff changeset
428 (Jan_1 : Day_Name;
kono
parents:
diff changeset
429 Year : Year_Number;
kono
parents:
diff changeset
430 Last_Year : Boolean := False;
kono
parents:
diff changeset
431 Next_Year : Boolean := False) return Day_Name;
kono
parents:
diff changeset
432 -- Given the weekday of January 1 in Year, determine the weekday on
kono
parents:
diff changeset
433 -- which January 1 fell last year or will fall next year as set by
kono
parents:
diff changeset
434 -- the two flags. This routine does not call Time_Of or Split.
kono
parents:
diff changeset
435
kono
parents:
diff changeset
436 function Last_Year_Has_53_Weeks
kono
parents:
diff changeset
437 (Jan_1 : Day_Name;
kono
parents:
diff changeset
438 Year : Year_Number) return Boolean;
kono
parents:
diff changeset
439 -- Given the weekday of January 1 in Year, determine whether last year
kono
parents:
diff changeset
440 -- has 53 weeks. A False value implies that the year has 52 weeks.
kono
parents:
diff changeset
441
kono
parents:
diff changeset
442 -------------
kono
parents:
diff changeset
443 -- Is_Leap --
kono
parents:
diff changeset
444 -------------
kono
parents:
diff changeset
445
kono
parents:
diff changeset
446 function Is_Leap (Year : Year_Number) return Boolean is
kono
parents:
diff changeset
447 begin
kono
parents:
diff changeset
448 if Year mod 400 = 0 then
kono
parents:
diff changeset
449 return True;
kono
parents:
diff changeset
450 elsif Year mod 100 = 0 then
kono
parents:
diff changeset
451 return False;
kono
parents:
diff changeset
452 else
kono
parents:
diff changeset
453 return Year mod 4 = 0;
kono
parents:
diff changeset
454 end if;
kono
parents:
diff changeset
455 end Is_Leap;
kono
parents:
diff changeset
456
kono
parents:
diff changeset
457 -----------------------
kono
parents:
diff changeset
458 -- Jan_1_Day_Of_Week --
kono
parents:
diff changeset
459 -----------------------
kono
parents:
diff changeset
460
kono
parents:
diff changeset
461 function Jan_1_Day_Of_Week
kono
parents:
diff changeset
462 (Jan_1 : Day_Name;
kono
parents:
diff changeset
463 Year : Year_Number;
kono
parents:
diff changeset
464 Last_Year : Boolean := False;
kono
parents:
diff changeset
465 Next_Year : Boolean := False) return Day_Name
kono
parents:
diff changeset
466 is
kono
parents:
diff changeset
467 Shift : Integer := 0;
kono
parents:
diff changeset
468
kono
parents:
diff changeset
469 begin
kono
parents:
diff changeset
470 if Last_Year then
kono
parents:
diff changeset
471 Shift := (if Is_Leap (Year - 1) then -2 else -1);
kono
parents:
diff changeset
472 elsif Next_Year then
kono
parents:
diff changeset
473 Shift := (if Is_Leap (Year) then 2 else 1);
kono
parents:
diff changeset
474 end if;
kono
parents:
diff changeset
475
kono
parents:
diff changeset
476 return Day_Name'Val ((Day_Name'Pos (Jan_1) + Shift) mod 7);
kono
parents:
diff changeset
477 end Jan_1_Day_Of_Week;
kono
parents:
diff changeset
478
kono
parents:
diff changeset
479 ----------------------------
kono
parents:
diff changeset
480 -- Last_Year_Has_53_Weeks --
kono
parents:
diff changeset
481 ----------------------------
kono
parents:
diff changeset
482
kono
parents:
diff changeset
483 function Last_Year_Has_53_Weeks
kono
parents:
diff changeset
484 (Jan_1 : Day_Name;
kono
parents:
diff changeset
485 Year : Year_Number) return Boolean
kono
parents:
diff changeset
486 is
kono
parents:
diff changeset
487 Last_Jan_1 : constant Day_Name :=
kono
parents:
diff changeset
488 Jan_1_Day_Of_Week (Jan_1, Year, Last_Year => True);
kono
parents:
diff changeset
489
kono
parents:
diff changeset
490 begin
kono
parents:
diff changeset
491 -- These two cases are illustrated in the table below
kono
parents:
diff changeset
492
kono
parents:
diff changeset
493 return
kono
parents:
diff changeset
494 Last_Jan_1 = Thursday
kono
parents:
diff changeset
495 or else (Last_Jan_1 = Wednesday and then Is_Leap (Year - 1));
kono
parents:
diff changeset
496 end Last_Year_Has_53_Weeks;
kono
parents:
diff changeset
497
kono
parents:
diff changeset
498 -- Start of processing for Week_In_Year
kono
parents:
diff changeset
499
kono
parents:
diff changeset
500 begin
kono
parents:
diff changeset
501 Split (Date, Year, Month, Day, Hour, Minute, Second, Sub_Second);
kono
parents:
diff changeset
502
kono
parents:
diff changeset
503 -- According to ISO 8601, the first week of year Y is the week that
kono
parents:
diff changeset
504 -- contains the first Thursday in year Y. The following table contains
kono
parents:
diff changeset
505 -- all possible combinations of years and weekdays along with examples.
kono
parents:
diff changeset
506
kono
parents:
diff changeset
507 -- +-------+------+-------+---------+
kono
parents:
diff changeset
508 -- | Jan 1 | Leap | Weeks | Example |
kono
parents:
diff changeset
509 -- +-------+------+-------+---------+
kono
parents:
diff changeset
510 -- | Mon | No | 52 | 2007 |
kono
parents:
diff changeset
511 -- +-------+------+-------+---------+
kono
parents:
diff changeset
512 -- | Mon | Yes | 52 | 1996 |
kono
parents:
diff changeset
513 -- +-------+------+-------+---------+
kono
parents:
diff changeset
514 -- | Tue | No | 52 | 2002 |
kono
parents:
diff changeset
515 -- +-------+------+-------+---------+
kono
parents:
diff changeset
516 -- | Tue | Yes | 52 | 1980 |
kono
parents:
diff changeset
517 -- +-------+------+-------+---------+
kono
parents:
diff changeset
518 -- | Wed | No | 52 | 2003 |
kono
parents:
diff changeset
519 -- +-------+------#########---------+
kono
parents:
diff changeset
520 -- | Wed | Yes # 53 # 1992 |
kono
parents:
diff changeset
521 -- +-------+------#-------#---------+
kono
parents:
diff changeset
522 -- | Thu | No # 53 # 1998 |
kono
parents:
diff changeset
523 -- +-------+------#-------#---------+
kono
parents:
diff changeset
524 -- | Thu | Yes # 53 # 2004 |
kono
parents:
diff changeset
525 -- +-------+------#########---------+
kono
parents:
diff changeset
526 -- | Fri | No | 52 | 1999 |
kono
parents:
diff changeset
527 -- +-------+------+-------+---------+
kono
parents:
diff changeset
528 -- | Fri | Yes | 52 | 1988 |
kono
parents:
diff changeset
529 -- +-------+------+-------+---------+
kono
parents:
diff changeset
530 -- | Sat | No | 52 | 1994 |
kono
parents:
diff changeset
531 -- +-------+------+-------+---------+
kono
parents:
diff changeset
532 -- | Sat | Yes | 52 | 1972 |
kono
parents:
diff changeset
533 -- +-------+------+-------+---------+
kono
parents:
diff changeset
534 -- | Sun | No | 52 | 1995 |
kono
parents:
diff changeset
535 -- +-------+------+-------+---------+
kono
parents:
diff changeset
536 -- | Sun | Yes | 52 | 1956 |
kono
parents:
diff changeset
537 -- +-------+------+-------+---------+
kono
parents:
diff changeset
538
kono
parents:
diff changeset
539 -- A small optimization, the input date is January 1. Note that this
kono
parents:
diff changeset
540 -- is a key day since it determines the number of weeks and is used
kono
parents:
diff changeset
541 -- when special casing the first week of January and the last week of
kono
parents:
diff changeset
542 -- December.
kono
parents:
diff changeset
543
kono
parents:
diff changeset
544 Jan_1 := Day_Of_Week (if Day = 1 and then Month = 1
kono
parents:
diff changeset
545 then Date
kono
parents:
diff changeset
546 else (Time_Of (Year, 1, 1, 0.0)));
kono
parents:
diff changeset
547
kono
parents:
diff changeset
548 -- Special cases for January
kono
parents:
diff changeset
549
kono
parents:
diff changeset
550 if Month = 1 then
kono
parents:
diff changeset
551
kono
parents:
diff changeset
552 -- Special case 1: January 1, 2 and 3. These three days may belong
kono
parents:
diff changeset
553 -- to last year's last week which can be week number 52 or 53.
kono
parents:
diff changeset
554
kono
parents:
diff changeset
555 -- +-----+-----+-----+=====+-----+-----+-----+
kono
parents:
diff changeset
556 -- | Mon | Tue | Wed # Thu # Fri | Sat | Sun |
kono
parents:
diff changeset
557 -- +-----+-----+-----+-----+-----+-----+-----+
kono
parents:
diff changeset
558 -- | 26 | 27 | 28 # 29 # 30 | 31 | 1 |
kono
parents:
diff changeset
559 -- +-----+-----+-----+-----+-----+-----+-----+
kono
parents:
diff changeset
560 -- | 27 | 28 | 29 # 30 # 31 | 1 | 2 |
kono
parents:
diff changeset
561 -- +-----+-----+-----+-----+-----+-----+-----+
kono
parents:
diff changeset
562 -- | 28 | 29 | 30 # 31 # 1 | 2 | 3 |
kono
parents:
diff changeset
563 -- +-----+-----+-----+=====+-----+-----+-----+
kono
parents:
diff changeset
564
kono
parents:
diff changeset
565 if (Day = 1 and then Jan_1 in Friday .. Sunday)
kono
parents:
diff changeset
566 or else
kono
parents:
diff changeset
567 (Day = 2 and then Jan_1 in Friday .. Saturday)
kono
parents:
diff changeset
568 or else
kono
parents:
diff changeset
569 (Day = 3 and then Jan_1 = Friday)
kono
parents:
diff changeset
570 then
kono
parents:
diff changeset
571 Week := (if Last_Year_Has_53_Weeks (Jan_1, Year) then 53 else 52);
kono
parents:
diff changeset
572
kono
parents:
diff changeset
573 -- January 1, 2 and 3 belong to the previous year
kono
parents:
diff changeset
574
kono
parents:
diff changeset
575 Year := Year - 1;
kono
parents:
diff changeset
576 return;
kono
parents:
diff changeset
577
kono
parents:
diff changeset
578 -- Special case 2: January 1, 2, 3, 4, 5, 6 and 7 of the first week
kono
parents:
diff changeset
579
kono
parents:
diff changeset
580 -- +-----+-----+-----+=====+-----+-----+-----+
kono
parents:
diff changeset
581 -- | Mon | Tue | Wed # Thu # Fri | Sat | Sun |
kono
parents:
diff changeset
582 -- +-----+-----+-----+-----+-----+-----+-----+
kono
parents:
diff changeset
583 -- | 29 | 30 | 31 # 1 # 2 | 3 | 4 |
kono
parents:
diff changeset
584 -- +-----+-----+-----+-----+-----+-----+-----+
kono
parents:
diff changeset
585 -- | 30 | 31 | 1 # 2 # 3 | 4 | 5 |
kono
parents:
diff changeset
586 -- +-----+-----+-----+-----+-----+-----+-----+
kono
parents:
diff changeset
587 -- | 31 | 1 | 2 # 3 # 4 | 5 | 6 |
kono
parents:
diff changeset
588 -- +-----+-----+-----+-----+-----+-----+-----+
kono
parents:
diff changeset
589 -- | 1 | 2 | 3 # 4 # 5 | 6 | 7 |
kono
parents:
diff changeset
590 -- +-----+-----+-----+=====+-----+-----+-----+
kono
parents:
diff changeset
591
kono
parents:
diff changeset
592 elsif (Day <= 4 and then Jan_1 in Monday .. Thursday)
kono
parents:
diff changeset
593 or else
kono
parents:
diff changeset
594 (Day = 5 and then Jan_1 in Monday .. Wednesday)
kono
parents:
diff changeset
595 or else
kono
parents:
diff changeset
596 (Day = 6 and then Jan_1 in Monday .. Tuesday)
kono
parents:
diff changeset
597 or else
kono
parents:
diff changeset
598 (Day = 7 and then Jan_1 = Monday)
kono
parents:
diff changeset
599 then
kono
parents:
diff changeset
600 Week := 1;
kono
parents:
diff changeset
601 return;
kono
parents:
diff changeset
602 end if;
kono
parents:
diff changeset
603
kono
parents:
diff changeset
604 -- Month other than 1
kono
parents:
diff changeset
605
kono
parents:
diff changeset
606 -- Special case 3: December 29, 30 and 31. These days may belong to
kono
parents:
diff changeset
607 -- next year's first week.
kono
parents:
diff changeset
608
kono
parents:
diff changeset
609 -- +-----+-----+-----+=====+-----+-----+-----+
kono
parents:
diff changeset
610 -- | Mon | Tue | Wed # Thu # Fri | Sat | Sun |
kono
parents:
diff changeset
611 -- +-----+-----+-----+-----+-----+-----+-----+
kono
parents:
diff changeset
612 -- | 29 | 30 | 31 # 1 # 2 | 3 | 4 |
kono
parents:
diff changeset
613 -- +-----+-----+-----+-----+-----+-----+-----+
kono
parents:
diff changeset
614 -- | 30 | 31 | 1 # 2 # 3 | 4 | 5 |
kono
parents:
diff changeset
615 -- +-----+-----+-----+-----+-----+-----+-----+
kono
parents:
diff changeset
616 -- | 31 | 1 | 2 # 3 # 4 | 5 | 6 |
kono
parents:
diff changeset
617 -- +-----+-----+-----+=====+-----+-----+-----+
kono
parents:
diff changeset
618
kono
parents:
diff changeset
619 elsif Month = 12 and then Day > 28 then
kono
parents:
diff changeset
620 declare
kono
parents:
diff changeset
621 Next_Jan_1 : constant Day_Name :=
kono
parents:
diff changeset
622 Jan_1_Day_Of_Week (Jan_1, Year, Next_Year => True);
kono
parents:
diff changeset
623 begin
kono
parents:
diff changeset
624 if (Day = 29 and then Next_Jan_1 = Thursday)
kono
parents:
diff changeset
625 or else
kono
parents:
diff changeset
626 (Day = 30 and then Next_Jan_1 in Wednesday .. Thursday)
kono
parents:
diff changeset
627 or else
kono
parents:
diff changeset
628 (Day = 31 and then Next_Jan_1 in Tuesday .. Thursday)
kono
parents:
diff changeset
629 then
kono
parents:
diff changeset
630 Year := Year + 1;
kono
parents:
diff changeset
631 Week := 1;
kono
parents:
diff changeset
632 return;
kono
parents:
diff changeset
633 end if;
kono
parents:
diff changeset
634 end;
kono
parents:
diff changeset
635 end if;
kono
parents:
diff changeset
636
kono
parents:
diff changeset
637 -- Determine the week from which to start counting. If January 1 does
kono
parents:
diff changeset
638 -- not belong to the first week of the input year, then the next week
kono
parents:
diff changeset
639 -- is the first week.
kono
parents:
diff changeset
640
kono
parents:
diff changeset
641 Start_Week := (if Jan_1 in Friday .. Sunday then 1 else 2);
kono
parents:
diff changeset
642
kono
parents:
diff changeset
643 -- At this point all special combinations have been accounted for and
kono
parents:
diff changeset
644 -- the proper start week has been found. Since January 1 may not fall
kono
parents:
diff changeset
645 -- on a Monday, shift 7 - Day_Name'Pos (Jan_1). This action ensures an
kono
parents:
diff changeset
646 -- origin which falls on Monday.
kono
parents:
diff changeset
647
kono
parents:
diff changeset
648 Shift := 7 - Day_Name'Pos (Jan_1);
kono
parents:
diff changeset
649 Week := Start_Week + (Day_In_Year (Date) - Shift - 1) / 7;
kono
parents:
diff changeset
650 end Year_Week_In_Year;
kono
parents:
diff changeset
651
kono
parents:
diff changeset
652 end GNAT.Calendar;