1 /*
2 ***********************************************************************
3 * Copyright (C) 2016 and later: Unicode, Inc. and others.
4 * License & terms of use: http://www.unicode.org/copyright.html#License
5 ***********************************************************************
6 **********************************************************************
7 * Copyright (C) 1998-2012, International Business Machines Corporation
8 * and others.  All Rights Reserved.
9 **********************************************************************
10 *
11 * File date.c
12 *
13 * Modification History:
14 *
15 *   Date        Name        Description
16 *   06/16/99    stephen     Creation.
17 *******************************************************************************
18 */
19 
20 #include <stdio.h>
21 #include <string.h>
22 #include <stdlib.h>
23 
24 #include "unicode/uloc.h"
25 #include "unicode/udat.h"
26 #include "unicode/ucal.h"
27 #include "unicode/ustring.h"
28 #include "unicode/uclean.h"
29 
30 #include "uprint.h"
31 
32 #if UCONFIG_NO_FORMATTING
33 
main(int argc,char ** argv)34 int main(int argc, char **argv)
35 {
36   printf("%s: Sorry, UCONFIG_NO_FORMATTING was turned on (see uconfig.h). No formatting can be done. \n", argv[0]);
37   return 0;
38 }
39 #else
40 
41 
42 /* Protos */
43 static void usage(void);
44 
45 static void version(void);
46 
47 static void cal(int32_t month, int32_t year,
48                 UBool useLongNames, UErrorCode *status);
49 
50 static void get_symbols(const UDateFormat *fmt,
51                         UDateFormatSymbolType type,
52                         UChar *array[],
53                         int32_t arrayLength,
54                         int32_t lowestIndex,
55                         int32_t firstIndex,
56                         UErrorCode *status);
57 
58 static void free_symbols(UChar *array[],
59                          int32_t arrayLength);
60 
61 static void get_days(const UDateFormat *fmt,
62                      UChar *days [], UBool useLongNames,
63                      int32_t fdow, UErrorCode *status);
64 
65 static void free_days(UChar *days[]);
66 
67 static void get_months(const UDateFormat *fmt,
68                        UChar *months [], UBool useLongNames,
69                        UErrorCode *status);
70 
71 static void free_months(UChar *months[]);
72 
73 static void indent(int32_t count, FILE *f);
74 
75 static void print_days(UChar *days [], FILE *f, UErrorCode *status);
76 
77 static void  print_month(UCalendar *c,
78                          UChar *days [],
79                          UBool useLongNames, int32_t fdow,
80                          UErrorCode *status);
81 
82 static void  print_year(UCalendar *c,
83                         UChar *days [], UChar *months [],
84                         UBool useLongNames, int32_t fdow,
85                         UErrorCode *status);
86 
87 /* The version of cal */
88 #define CAL_VERSION "1.0"
89 
90 /* Number of days in a week */
91 #define DAY_COUNT 7
92 
93 /* Number of months in a year (yes, 13) */
94 #define MONTH_COUNT 13
95 
96 /* Separation between months in year view */
97 #define MARGIN_WIDTH 4
98 
99 /* Size of stack buffers */
100 #define BUF_SIZE 64
101 
102 /* Patterm string - "MMM yyyy" */
103 static const UChar sShortPat [] = { 0x004D, 0x004D, 0x004D, 0x0020,
104 0x0079, 0x0079, 0x0079, 0x0079 };
105 /* Pattern string - "MMMM yyyy" */
106 static const UChar sLongPat [] = { 0x004D, 0x004D, 0x004D, 0x004D, 0x0020,
107 0x0079, 0x0079, 0x0079, 0x0079 };
108 
109 
110 int
main(int argc,char ** argv)111 main(int argc,
112      char **argv)
113 {
114     int printUsage = 0;
115     int printVersion = 0;
116     UBool useLongNames = 0;
117     int optInd = 1;
118     char *arg;
119     int32_t month = -1, year = -1;
120     UErrorCode status = U_ZERO_ERROR;
121 
122 
123     /* parse the options */
124     for(optInd = 1; optInd < argc; ++optInd) {
125         arg = argv[optInd];
126 
127         /* version info */
128         if(strcmp(arg, "-v") == 0 || strcmp(arg, "--version") == 0) {
129             printVersion = 1;
130         }
131         /* usage info */
132         else if(strcmp(arg, "-h") == 0 || strcmp(arg, "--help") == 0) {
133             printUsage = 1;
134         }
135         /* use long day names */
136         else if(strcmp(arg, "-l") == 0 || strcmp(arg, "--long") == 0) {
137             useLongNames = 1;
138         }
139         /* POSIX.1 says all arguments after -- are not options */
140         else if(strcmp(arg, "--") == 0) {
141             /* skip the -- */
142             ++optInd;
143             break;
144         }
145         /* unrecognized option */
146         else if(strncmp(arg, "-", strlen("-")) == 0) {
147             printf("cal: invalid option -- %s\n", arg+1);
148             printUsage = 1;
149         }
150         /* done with options, display cal */
151         else {
152             break;
153         }
154     }
155 
156     /* Get the month and year to display, if specified */
157     if(optInd != argc) {
158 
159         /* Month and year specified */
160         if(argc - optInd == 2) {
161             sscanf(argv[optInd], "%d", (int*)&month);
162             sscanf(argv[optInd + 1], "%d", (int*)&year);
163 
164             /* Make sure the month value is legal */
165             if(month < 0 || month > 12) {
166                 printf("icucal: Bad value for month -- %d\n", (int)month);
167 
168                 /* Display usage */
169                 printUsage = 1;
170             }
171 
172             /* Adjust because months are 0-based */
173             --month;
174         }
175         /* Only year specified */
176         else {
177             sscanf(argv[optInd], "%d", (int*)&year);
178         }
179     }
180 
181     /* print usage info */
182     if(printUsage) {
183         usage();
184         return 0;
185     }
186 
187     /* print version info */
188     if(printVersion) {
189         version();
190         return 0;
191     }
192 
193     /* print the cal */
194     cal(month, year, useLongNames, &status);
195 
196     /* ICU cleanup.  Deallocate any memory ICU may be holding.  */
197     u_cleanup();
198 
199     return (U_FAILURE(status) ? 1 : 0);
200 }
201 
202 /* Usage information */
203 static void
usage()204 usage()
205 {
206     puts("Usage: icucal [OPTIONS] [[MONTH] YEAR]");
207     puts("");
208     puts("Options:");
209     puts("  -h, --help        Print this message and exit.");
210     puts("  -v, --version     Print the version number of cal and exit.");
211     puts("  -l, --long        Use long names.");
212     puts("");
213     puts("Arguments (optional):");
214     puts("  MONTH             An integer (1-12) indicating the month to display");
215     puts("  YEAR              An integer indicating the year to display");
216     puts("");
217     puts("For an interesting calendar, look at October 1582");
218 }
219 
220 /* Version information */
221 static void
version()222 version()
223 {
224     printf("icucal version %s (ICU version %s), created by Stephen F. Booth.\n",
225         CAL_VERSION, U_ICU_VERSION);
226     puts(U_COPYRIGHT_STRING);
227 }
228 
229 static void
cal(int32_t month,int32_t year,UBool useLongNames,UErrorCode * status)230 cal(int32_t month,
231     int32_t year,
232     UBool useLongNames,
233     UErrorCode *status)
234 {
235     UCalendar *c;
236     UChar *days [DAY_COUNT];
237     UChar *months [MONTH_COUNT];
238     int32_t fdow;
239 
240     if(U_FAILURE(*status)) return;
241 
242     /* Create a new calendar */
243     c = ucal_open(0, -1, uloc_getDefault(), UCAL_TRADITIONAL, status);
244 
245     /* Determine if we are printing a calendar for one month or for a year */
246 
247     /* Print an entire year */
248     if(month == -1 && year != -1) {
249 
250         /* Set the year */
251         ucal_set(c, UCAL_YEAR, year);
252 
253         /* Determine the first day of the week */
254         fdow = ucal_getAttribute(c, UCAL_FIRST_DAY_OF_WEEK);
255 
256         /* Print the calendar for the year */
257         print_year(c, days, months, useLongNames, fdow, status);
258     }
259 
260     /* Print only one month */
261     else {
262 
263         /* Set the month and the year, if specified */
264         if(month != -1)
265             ucal_set(c, UCAL_MONTH, month);
266         if(year != -1)
267             ucal_set(c, UCAL_YEAR, year);
268 
269         /* Determine the first day of the week */
270         fdow = ucal_getAttribute(c, UCAL_FIRST_DAY_OF_WEEK);
271 
272         /* Print the calendar for the month */
273         print_month(c, days, useLongNames, fdow, status);
274     }
275 
276     /* Clean up */
277     ucal_close(c);
278 }
279 /*
280  * Get a set of DateFormat symbols of a given type.
281  *
282  * lowestIndex is the index of the first symbol to fetch.
283  * (e.g. it will be one to fetch day names, since Sunday is
284  *  day 1 *not* day 0.)
285  *
286  * firstIndex is the index of the symbol to place first in
287  * the output array. This is used when fetching day names
288  * in locales where the week doesn't start on Sunday.
289  */
get_symbols(const UDateFormat * fmt,UDateFormatSymbolType type,UChar * array[],int32_t arrayLength,int32_t lowestIndex,int32_t firstIndex,UErrorCode * status)290 static void get_symbols(const UDateFormat *fmt,
291                         UDateFormatSymbolType type,
292                         UChar *array[],
293                         int32_t arrayLength,
294                         int32_t lowestIndex,
295                         int32_t firstIndex,
296                         UErrorCode *status)
297 {
298     int32_t count, i;
299 
300     if (U_FAILURE(*status)) {
301         return;
302     }
303 
304     count = udat_countSymbols(fmt, type);
305 
306     if(count != arrayLength + lowestIndex) {
307         return;
308     }
309 
310     for(i = 0; i < arrayLength; i++) {
311         int32_t idx = (i + firstIndex) % arrayLength;
312         int32_t size = 1 + udat_getSymbols(fmt, type, idx + lowestIndex, NULL, 0, status);
313 
314         array[idx] = (UChar *) malloc(sizeof(UChar) * size);
315 
316         *status = U_ZERO_ERROR;
317         udat_getSymbols(fmt, type, idx + lowestIndex, array[idx], size, status);
318     }
319 }
320 
321 /* Free the symbols allocated by get_symbols(). */
free_symbols(UChar * array[],int32_t arrayLength)322 static void free_symbols(UChar *array[],
323                          int32_t arrayLength)
324 {
325     int32_t i;
326 
327     for(i = 0; i < arrayLength; i++) {
328         free(array[i]);
329     }
330 }
331 
332 /* Get the day names for the specified locale, in either long or short
333 form.  Also, reorder the days so that they are in the proper order
334 for the locale (not all locales begin weeks on Sunday; in France,
335 weeks start on Monday) */
336 static void
get_days(const UDateFormat * fmt,UChar * days[],UBool useLongNames,int32_t fdow,UErrorCode * status)337 get_days(const UDateFormat *fmt,
338          UChar *days [],
339          UBool useLongNames,
340          int32_t fdow,
341          UErrorCode *status)
342 {
343     UDateFormatSymbolType dayType = (useLongNames ? UDAT_WEEKDAYS : UDAT_SHORT_WEEKDAYS);
344 
345     if(U_FAILURE(*status))
346         return;
347 
348     /* fdow is 1-based */
349     --fdow;
350 
351     get_symbols(fmt, dayType, days, DAY_COUNT, 1, fdow, status);
352 }
353 
free_days(UChar * days[])354 static void free_days(UChar *days[])
355 {
356     free_symbols(days, DAY_COUNT);
357 }
358 
359 /* Get the month names for the specified locale, in either long or
360 short form. */
361 static void
get_months(const UDateFormat * fmt,UChar * months[],UBool useLongNames,UErrorCode * status)362 get_months(const UDateFormat *fmt,
363            UChar *months [],
364            UBool useLongNames,
365            UErrorCode *status)
366 {
367     UDateFormatSymbolType monthType = (useLongNames ? UDAT_MONTHS : UDAT_SHORT_MONTHS);
368 
369     if(U_FAILURE(*status))
370         return;
371 
372     get_symbols(fmt, monthType, months, MONTH_COUNT - 1, 0, 0, status); /* some locales have 13 months, no idea why */
373 }
374 
free_months(UChar * months[])375 static void free_months(UChar *months[])
376 {
377     free_symbols(months, MONTH_COUNT - 1);
378 }
379 
380 /* Indent a certain number of spaces */
381 static void
indent(int32_t count,FILE * f)382 indent(int32_t count,
383        FILE *f)
384 {
385     char c [BUF_SIZE];
386 
387     if(count <= 0)
388     {
389         return;
390     }
391 
392     if(count < BUF_SIZE) {
393         memset(c, (int)' ', count);
394         fwrite(c, sizeof(char), count, f);
395     }
396     else {
397         int32_t i;
398         for(i = 0; i < count; ++i)
399             putc(' ', f);
400     }
401 }
402 
403 /* Print the days */
404 static void
print_days(UChar * days[],FILE * f,UErrorCode * status)405 print_days(UChar *days [],
406            FILE *f,
407            UErrorCode *status)
408 {
409     int32_t i;
410 
411     if(U_FAILURE(*status)) return;
412 
413     /* Print the day names */
414     for(i = 0; i < DAY_COUNT; ++i) {
415         uprint(days[i], f, status);
416         putc(' ', f);
417     }
418 }
419 
420 /* Print out a calendar for c's current month */
421 static void
print_month(UCalendar * c,UChar * days[],UBool useLongNames,int32_t fdow,UErrorCode * status)422 print_month(UCalendar *c,
423             UChar *days [],
424             UBool useLongNames,
425             int32_t fdow,
426             UErrorCode *status)
427 {
428     int32_t width, pad, i, day;
429     int32_t lens [DAY_COUNT];
430     int32_t firstday, current;
431     UNumberFormat *nfmt;
432     UDateFormat *dfmt;
433     UChar s [BUF_SIZE];
434     const UChar *pat = (useLongNames ? sLongPat : sShortPat);
435     int32_t len = (useLongNames ? 9 : 8);
436 
437     if(U_FAILURE(*status)) return;
438 
439 
440     /* ========== Generate the header containing the month and year */
441 
442     /* Open a formatter with a month and year only pattern */
443     dfmt = udat_open(UDAT_PATTERN,UDAT_PATTERN,NULL,NULL,0,pat, len,status);
444 
445     /* Format the date */
446     udat_format(dfmt, ucal_getMillis(c, status), s, BUF_SIZE, 0, status);
447 
448 
449     /* ========== Get the day names */
450     get_days(dfmt, days, useLongNames, fdow, status);
451 
452     /* ========== Print the header */
453 
454     /* Calculate widths for justification */
455     width = 6; /* 6 spaces, 1 between each day name */
456     for(i = 0; i < DAY_COUNT; ++i) {
457         lens[i] = u_strlen(days[i]);
458         width += lens[i];
459     }
460 
461     /* Print the header, centered among the day names */
462     pad = width - u_strlen(s);
463     indent(pad / 2, stdout);
464     uprint(s, stdout, status);
465     putc('\n', stdout);
466 
467 
468     /* ========== Print the day names */
469 
470     print_days(days, stdout, status);
471     putc('\n', stdout);
472 
473 
474     /* ========== Print the calendar */
475 
476     /* Get the first of the month */
477     ucal_set(c, UCAL_DATE, 1);
478     firstday = ucal_get(c, UCAL_DAY_OF_WEEK, status);
479 
480     /* The day of the week for the first day of the month is based on
481     1-based days of the week, which were also reordered when placed
482     in the days array.  Account for this here by offsetting by the
483     first day of the week for the locale, which is also 1-based. */
484     firstday -= fdow;
485 
486     /* Open the formatter */
487     nfmt = unum_open(UNUM_DECIMAL, NULL,0,NULL,NULL, status);
488 
489     /* Indent the correct number of spaces for the first week */
490     current = firstday;
491     if(current < 0)
492     {
493        current += 7;
494     }
495     for(i = 0; i < current; ++i)
496         indent(lens[i] + 1, stdout);
497 
498     /* Finally, print out the days */
499     day = ucal_get(c, UCAL_DATE, status);
500     do {
501 
502         /* Format the current day string */
503         unum_format(nfmt, day, s, BUF_SIZE, 0, status);
504 
505         /* Calculate the justification and indent */
506         pad = lens[current] - u_strlen(s);
507         indent(pad, stdout);
508 
509         /* Print the day number out, followed by a space */
510         uprint(s, stdout, status);
511         putc(' ', stdout);
512 
513         /* Update the current day */
514         ++current;
515         current %= DAY_COUNT;
516 
517         /* If we're at day 0 (first day of the week), insert a newline */
518         if(current == 0) {
519             putc('\n', stdout);
520         }
521 
522         /* Go to the next day */
523         ucal_add(c, UCAL_DATE, 1, status);
524         day = ucal_get(c, UCAL_DATE, status);
525 
526     } while(day != 1);
527 
528     /* Output a trailing newline */
529     putc('\n', stdout);
530 
531     /* Clean up */
532     free_days(days);
533     unum_close(nfmt);
534     udat_close(dfmt);
535 }
536 
537 /* Print out a calendar for c's current year */
538 static void
print_year(UCalendar * c,UChar * days[],UChar * months[],UBool useLongNames,int32_t fdow,UErrorCode * status)539 print_year(UCalendar *c,
540            UChar *days [],
541            UChar *months [],
542            UBool useLongNames,
543            int32_t fdow,
544            UErrorCode *status)
545 {
546     int32_t width, pad, i, j;
547     int32_t lens [DAY_COUNT];
548     UNumberFormat *nfmt;
549     UDateFormat *dfmt;
550     UChar s [BUF_SIZE];
551     const UChar pat [] = { 0x0079, 0x0079, 0x0079, 0x0079 };
552     int32_t len = 4;
553     UCalendar  *left_cal, *right_cal;
554     int32_t left_day, right_day;
555     int32_t left_firstday, right_firstday, left_current, right_current;
556     int32_t left_month, right_month;
557 
558     if(U_FAILURE(*status)) return;
559 
560     /* Alias */
561     left_cal = c;
562 
563     /* ========== Generate the header containing the year (only) */
564 
565     /* Open a formatter with a month and year only pattern */
566     dfmt = udat_open(UDAT_PATTERN,UDAT_PATTERN,NULL,NULL,0,pat, len, status);
567 
568     /* Format the date */
569     udat_format(dfmt, ucal_getMillis(left_cal, status), s, BUF_SIZE, 0, status);
570 
571     /* ========== Get the month and day names */
572     get_days(dfmt, days, useLongNames, fdow, status);
573     get_months(dfmt, months, useLongNames, status);
574 
575     /* ========== Print the header, centered */
576 
577     /* Calculate widths for justification */
578     width = 6; /* 6 spaces, 1 between each day name */
579     for(i = 0; i < DAY_COUNT; ++i) {
580         lens[i] = u_strlen(days[i]);
581         width += lens[i];
582     }
583 
584     /* width is the width for 1 calendar; we are displaying in 2 cols
585     with MARGIN_WIDTH spaces between months */
586 
587     /* Print the header, centered among the day names */
588     pad = 2 * width + MARGIN_WIDTH - u_strlen(s);
589     indent(pad / 2, stdout);
590     uprint(s, stdout, status);
591     putc('\n', stdout);
592     putc('\n', stdout);
593 
594     /* Generate a copy of the calendar to use */
595     right_cal = ucal_open(0, -1, uloc_getDefault(), UCAL_TRADITIONAL, status);
596     ucal_setMillis(right_cal, ucal_getMillis(left_cal, status), status);
597 
598     /* Open the formatter */
599     nfmt = unum_open(UNUM_DECIMAL,NULL, 0,NULL,NULL, status);
600 
601     /* ========== Calculate and display the months, two at a time */
602     for(i = 0; i < MONTH_COUNT - 1; i += 2) {
603 
604         /* Print the month names for the two current months */
605         pad = width - u_strlen(months[i]);
606         indent(pad / 2, stdout);
607         uprint(months[i], stdout, status);
608         indent(pad / 2 + MARGIN_WIDTH, stdout);
609         pad = width - u_strlen(months[i + 1]);
610         indent(pad / 2, stdout);
611         uprint(months[i + 1], stdout, status);
612         putc('\n', stdout);
613 
614         /* Print the day names, twice  */
615         print_days(days, stdout, status);
616         indent(MARGIN_WIDTH, stdout);
617         print_days(days, stdout, status);
618         putc('\n', stdout);
619 
620         /* Setup the two calendars */
621         ucal_set(left_cal, UCAL_MONTH, i);
622         ucal_set(left_cal, UCAL_DATE, 1);
623         ucal_set(right_cal, UCAL_MONTH, i + 1);
624         ucal_set(right_cal, UCAL_DATE, 1);
625 
626         left_firstday = ucal_get(left_cal, UCAL_DAY_OF_WEEK, status);
627         right_firstday = ucal_get(right_cal, UCAL_DAY_OF_WEEK, status);
628 
629         /* The day of the week for the first day of the month is based on
630         1-based days of the week.  However, the days were reordered
631         when placed in the days array.  Account for this here by
632         offsetting by the first day of the week for the locale, which
633         is also 1-based. */
634 
635         /* We need to mod by DAY_COUNT since fdow can be > firstday.  IE,
636         if fdow = 2 = Monday (like in France) and the first day of the
637         month is a 1 = Sunday, we want firstday to be 6, not -1 */
638         left_firstday += (DAY_COUNT - fdow);
639         left_firstday %= DAY_COUNT;
640 
641         right_firstday += (DAY_COUNT - fdow);
642         right_firstday %= DAY_COUNT;
643 
644         left_current = left_firstday;
645         right_current = right_firstday;
646 
647         left_day = ucal_get(left_cal, UCAL_DATE, status);
648         right_day = ucal_get(right_cal, UCAL_DATE, status);
649 
650         left_month = ucal_get(left_cal, UCAL_MONTH, status);
651         right_month = ucal_get(right_cal, UCAL_MONTH, status);
652 
653         /* Finally, print out the days */
654         while(left_month == i || right_month == i + 1) {
655 
656         /* If the left month is finished printing, but the right month
657         still has days to be printed, indent the width of the days
658             strings and reset the left calendar's current day to 0 */
659             if(left_month != i && right_month == i + 1) {
660                 indent(width + 1, stdout);
661                 left_current = 0;
662             }
663 
664             while(left_month == i) {
665 
666             /* If the day is the first, indent the correct number of
667                 spaces for the first week */
668                 if(left_day == 1) {
669                     for(j = 0; j < left_current; ++j)
670                         indent(lens[j] + 1, stdout);
671                 }
672 
673                 /* Format the current day string */
674                 unum_format(nfmt, left_day, s, BUF_SIZE, 0, status);
675 
676                 /* Calculate the justification and indent */
677                 pad = lens[left_current] - u_strlen(s);
678                 indent(pad, stdout);
679 
680                 /* Print the day number out, followed by a space */
681                 uprint(s, stdout, status);
682                 putc(' ', stdout);
683 
684                 /* Update the current day */
685                 ++left_current;
686                 left_current %= DAY_COUNT;
687 
688                 /* Go to the next day */
689                 ucal_add(left_cal, UCAL_DATE, 1, status);
690                 left_day = ucal_get(left_cal, UCAL_DATE, status);
691 
692                 /* Determine the month */
693                 left_month = ucal_get(left_cal, UCAL_MONTH, status);
694 
695                 /* If we're at day 0 (first day of the week), break and go to
696                 the next month */
697                 if(left_current == 0) {
698                     break;
699                 }
700             };
701 
702             /* If the current day isn't 0, indent to make up for missing
703             days at the end of the month */
704             if(left_current != 0) {
705                 for(j = left_current; j < DAY_COUNT; ++j)
706                     indent(lens[j] + 1, stdout);
707             }
708 
709             /* Indent between the two months */
710             indent(MARGIN_WIDTH, stdout);
711 
712             while(right_month == i + 1) {
713 
714             /* If the day is the first, indent the correct number of
715                 spaces for the first week */
716                 if(right_day == 1) {
717                     for(j = 0; j < right_current; ++j)
718                         indent(lens[j] + 1, stdout);
719                 }
720 
721                 /* Format the current day string */
722                 unum_format(nfmt, right_day, s, BUF_SIZE, 0, status);
723 
724                 /* Calculate the justification and indent */
725                 pad = lens[right_current] - u_strlen(s);
726                 indent(pad, stdout);
727 
728                 /* Print the day number out, followed by a space */
729                 uprint(s, stdout, status);
730                 putc(' ', stdout);
731 
732                 /* Update the current day */
733                 ++right_current;
734                 right_current %= DAY_COUNT;
735 
736                 /* Go to the next day */
737                 ucal_add(right_cal, UCAL_DATE, 1, status);
738                 right_day = ucal_get(right_cal, UCAL_DATE, status);
739 
740                 /* Determine the month */
741                 right_month = ucal_get(right_cal, UCAL_MONTH, status);
742 
743                 /* If we're at day 0 (first day of the week), break out */
744                 if(right_current == 0) {
745                     break;
746                 }
747 
748             };
749 
750             /* Output a newline */
751             putc('\n', stdout);
752         }
753 
754         /* Output a trailing newline */
755         putc('\n', stdout);
756   }
757 
758   /* Clean up */
759   free_months(months);
760   free_days(days);
761   udat_close(dfmt);
762   unum_close(nfmt);
763   ucal_close(right_cal);
764 }
765 
766 #endif
767