xref: /dflybsd-src/usr.bin/localedef/time.c (revision d50f9ae3448247db98eb135b85b2a32e6e4187f4)
1cd1c6085SJohn Marino /*
2cd1c6085SJohn Marino  * Copyright 2010 Nexenta Systems, Inc.  All rights reserved.
3cd1c6085SJohn Marino  * Copyright 2015 John Marino <draco@marino.st>
48aa2b98bSJohn Marino  *
58aa2b98bSJohn Marino  * This source code is derived from the illumos localedef command, and
68aa2b98bSJohn Marino  * provided under BSD-style license terms by Nexenta Systems, Inc.
78aa2b98bSJohn Marino  *
88aa2b98bSJohn Marino  * Redistribution and use in source and binary forms, with or without
98aa2b98bSJohn Marino  * modification, are permitted provided that the following conditions
108aa2b98bSJohn Marino  * are met:
118aa2b98bSJohn Marino  *
128aa2b98bSJohn Marino  * 1. Redistributions of source code must retain the above copyright
138aa2b98bSJohn Marino  *    notice, this list of conditions and the following disclaimer.
148aa2b98bSJohn Marino  * 2. Redistributions in binary form must reproduce the above copyright
158aa2b98bSJohn Marino  *    notice, this list of conditions and the following disclaimer in the
168aa2b98bSJohn Marino  *    documentation and/or other materials provided with the distribution.
178aa2b98bSJohn Marino  *
188aa2b98bSJohn Marino  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
198aa2b98bSJohn Marino  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
208aa2b98bSJohn Marino  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
218aa2b98bSJohn Marino  * ARE DISCLAIMED.  IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
228aa2b98bSJohn Marino  * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
238aa2b98bSJohn Marino  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
248aa2b98bSJohn Marino  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
258aa2b98bSJohn Marino  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
268aa2b98bSJohn Marino  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
278aa2b98bSJohn Marino  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
288aa2b98bSJohn Marino  * POSSIBILITY OF SUCH DAMAGE.
29cd1c6085SJohn Marino  */
30cd1c6085SJohn Marino 
31cd1c6085SJohn Marino /*
32cd1c6085SJohn Marino  * LC_TIME database generation routines for localedef.
33cd1c6085SJohn Marino  */
34cd1c6085SJohn Marino 
35cd1c6085SJohn Marino #include <stdio.h>
36cd1c6085SJohn Marino #include <stdlib.h>
37cd1c6085SJohn Marino #include <errno.h>
38cd1c6085SJohn Marino #include <sys/types.h>
39cd1c6085SJohn Marino #include <string.h>
40cd1c6085SJohn Marino #include <unistd.h>
41cd1c6085SJohn Marino #include "localedef.h"
42cd1c6085SJohn Marino #include "parser.h"
43cd1c6085SJohn Marino #include "timelocal.h"
44cd1c6085SJohn Marino 
45*d50f9ae3SSascha Wildner static struct lc_time_T tm;
46cd1c6085SJohn Marino 
47cd1c6085SJohn Marino void
init_time(void)48cd1c6085SJohn Marino init_time(void)
49cd1c6085SJohn Marino {
50cd1c6085SJohn Marino 	(void) memset(&tm, 0, sizeof (tm));
51cd1c6085SJohn Marino }
52cd1c6085SJohn Marino 
53cd1c6085SJohn Marino void
add_time_str(wchar_t * wcs)54cd1c6085SJohn Marino add_time_str(wchar_t *wcs)
55cd1c6085SJohn Marino {
56cd1c6085SJohn Marino 	char	*str;
57cd1c6085SJohn Marino 
58cd1c6085SJohn Marino 	if ((str = to_mb_string(wcs)) == NULL) {
59cd1c6085SJohn Marino 		INTERR;
60cd1c6085SJohn Marino 		return;
61cd1c6085SJohn Marino 	}
62cd1c6085SJohn Marino 	free(wcs);
63cd1c6085SJohn Marino 
64cd1c6085SJohn Marino 	switch (last_kw) {
65cd1c6085SJohn Marino 	case T_D_T_FMT:
66cd1c6085SJohn Marino 		tm.c_fmt = str;
67cd1c6085SJohn Marino 		break;
68cd1c6085SJohn Marino 	case T_D_FMT:
69cd1c6085SJohn Marino 		tm.x_fmt = str;
70cd1c6085SJohn Marino 		break;
71cd1c6085SJohn Marino 	case T_T_FMT:
72cd1c6085SJohn Marino 		tm.X_fmt = str;
73cd1c6085SJohn Marino 		break;
74cd1c6085SJohn Marino 	case T_T_FMT_AMPM:
75cd1c6085SJohn Marino 		tm.ampm_fmt = str;
76cd1c6085SJohn Marino 		break;
77cd1c6085SJohn Marino 	case T_DATE_FMT:
78cd1c6085SJohn Marino 		/*
79cd1c6085SJohn Marino 		 * This one is a Solaris extension, Too bad date just
80cd1c6085SJohn Marino 		 * doesn't use %c, which would be simpler.
81cd1c6085SJohn Marino 		 */
82cd1c6085SJohn Marino 		tm.date_fmt = str;
83cd1c6085SJohn Marino 		break;
84cd1c6085SJohn Marino 	case T_ERA_D_FMT:
85cd1c6085SJohn Marino 	case T_ERA_T_FMT:
86cd1c6085SJohn Marino 	case T_ERA_D_T_FMT:
87cd1c6085SJohn Marino 		/* Silently ignore it. */
88cd1c6085SJohn Marino 		break;
89cd1c6085SJohn Marino 	default:
90cd1c6085SJohn Marino 		free(str);
91cd1c6085SJohn Marino 		INTERR;
92cd1c6085SJohn Marino 		break;
93cd1c6085SJohn Marino 	}
94cd1c6085SJohn Marino }
95cd1c6085SJohn Marino 
96cd1c6085SJohn Marino static void
add_list(const char * ptr[],char * str,int limit)97cd1c6085SJohn Marino add_list(const char *ptr[], char *str, int limit)
98cd1c6085SJohn Marino {
99cd1c6085SJohn Marino 	int	i;
100cd1c6085SJohn Marino 	for (i = 0; i < limit; i++) {
101cd1c6085SJohn Marino 		if (ptr[i] == NULL) {
102cd1c6085SJohn Marino 			ptr[i] = str;
103cd1c6085SJohn Marino 			return;
104cd1c6085SJohn Marino 		}
105cd1c6085SJohn Marino 	}
106cd1c6085SJohn Marino 	fprintf(stderr,"too many list elements");
107cd1c6085SJohn Marino }
108cd1c6085SJohn Marino 
109cd1c6085SJohn Marino void
add_time_list(wchar_t * wcs)110cd1c6085SJohn Marino add_time_list(wchar_t *wcs)
111cd1c6085SJohn Marino {
112cd1c6085SJohn Marino 	char *str;
113cd1c6085SJohn Marino 
114cd1c6085SJohn Marino 	if ((str = to_mb_string(wcs)) == NULL) {
115cd1c6085SJohn Marino 		INTERR;
116cd1c6085SJohn Marino 		return;
117cd1c6085SJohn Marino 	}
118cd1c6085SJohn Marino 	free(wcs);
119cd1c6085SJohn Marino 
120cd1c6085SJohn Marino 	switch (last_kw) {
121cd1c6085SJohn Marino 	case T_ABMON:
122cd1c6085SJohn Marino 		add_list(tm.mon, str, 12);
123cd1c6085SJohn Marino 		break;
124cd1c6085SJohn Marino 	case T_MON:
125cd1c6085SJohn Marino 		add_list(tm.month, str, 12);
126cd1c6085SJohn Marino 		break;
127cd1c6085SJohn Marino 	case T_ABDAY:
128cd1c6085SJohn Marino 		add_list(tm.wday, str, 7);
129cd1c6085SJohn Marino 		break;
130cd1c6085SJohn Marino 	case T_DAY:
131cd1c6085SJohn Marino 		add_list(tm.weekday, str, 7);
132cd1c6085SJohn Marino 		break;
133cd1c6085SJohn Marino 	case T_AM_PM:
134cd1c6085SJohn Marino 		if (tm.am == NULL) {
135cd1c6085SJohn Marino 			tm.am = str;
136cd1c6085SJohn Marino 		} else if (tm.pm == NULL) {
137cd1c6085SJohn Marino 			tm.pm = str;
138cd1c6085SJohn Marino 		} else {
139cd1c6085SJohn Marino 			fprintf(stderr,"too many list elements");
140cd1c6085SJohn Marino 		}
141cd1c6085SJohn Marino 		break;
142cd1c6085SJohn Marino 	case T_ALT_DIGITS:
143cd1c6085SJohn Marino 	case T_ERA:
144cd1c6085SJohn Marino 		free(str);
145cd1c6085SJohn Marino 		break;
146cd1c6085SJohn Marino 	default:
147cd1c6085SJohn Marino 		free(str);
148cd1c6085SJohn Marino 		INTERR;
149cd1c6085SJohn Marino 		break;
150cd1c6085SJohn Marino 	}
151cd1c6085SJohn Marino }
152cd1c6085SJohn Marino 
153cd1c6085SJohn Marino void
check_time_list(void)154cd1c6085SJohn Marino check_time_list(void)
155cd1c6085SJohn Marino {
156cd1c6085SJohn Marino 	switch (last_kw) {
157cd1c6085SJohn Marino 	case T_ABMON:
158cd1c6085SJohn Marino 		if (tm.mon[11] != NULL)
159cd1c6085SJohn Marino 			return;
160cd1c6085SJohn Marino 		break;
161cd1c6085SJohn Marino 	case T_MON:
162cd1c6085SJohn Marino 		if (tm.month[11] != NULL)
163cd1c6085SJohn Marino 			return;
164cd1c6085SJohn Marino 		break;
165cd1c6085SJohn Marino 	case T_ABDAY:
166cd1c6085SJohn Marino 		if (tm.wday[6] != NULL)
167cd1c6085SJohn Marino 			return;
168cd1c6085SJohn Marino 		break;
169cd1c6085SJohn Marino 	case T_DAY:
170cd1c6085SJohn Marino 		if (tm.weekday[6] != NULL)
171cd1c6085SJohn Marino 			return;
172cd1c6085SJohn Marino 		break;
173cd1c6085SJohn Marino 	case T_AM_PM:
174cd1c6085SJohn Marino 		if (tm.pm != NULL)
175cd1c6085SJohn Marino 			return;
176cd1c6085SJohn Marino 		break;
177cd1c6085SJohn Marino 	case T_ERA:
178cd1c6085SJohn Marino 	case T_ALT_DIGITS:
179cd1c6085SJohn Marino 		return;
180cd1c6085SJohn Marino 	default:
181cd1c6085SJohn Marino 		fprintf(stderr,"unknown list");
182cd1c6085SJohn Marino 		break;
183cd1c6085SJohn Marino 	}
184cd1c6085SJohn Marino 
185cd1c6085SJohn Marino 	fprintf(stderr,"too few items in list (%d)", last_kw);
186cd1c6085SJohn Marino }
187cd1c6085SJohn Marino 
188cd1c6085SJohn Marino #pragma GCC diagnostic push
189cd1c6085SJohn Marino #pragma GCC diagnostic ignored "-Wcast-qual"
190cd1c6085SJohn Marino 
191cd1c6085SJohn Marino void
reset_time_list(void)192cd1c6085SJohn Marino reset_time_list(void)
193cd1c6085SJohn Marino {
194cd1c6085SJohn Marino 	int i;
195cd1c6085SJohn Marino 	switch (last_kw) {
196cd1c6085SJohn Marino 	case T_ABMON:
197cd1c6085SJohn Marino 		for (i = 0; i < 12; i++) {
198cd1c6085SJohn Marino 			free((char *)tm.mon[i]);
199cd1c6085SJohn Marino 			tm.mon[i] = NULL;
200cd1c6085SJohn Marino 		}
201cd1c6085SJohn Marino 		break;
202cd1c6085SJohn Marino 	case T_MON:
203cd1c6085SJohn Marino 		for (i = 0; i < 12; i++) {
204cd1c6085SJohn Marino 			free((char *)tm.month[i]);
205cd1c6085SJohn Marino 			tm.month[i] = NULL;
206cd1c6085SJohn Marino 		}
207cd1c6085SJohn Marino 		break;
208cd1c6085SJohn Marino 	case T_ABDAY:
209cd1c6085SJohn Marino 		for (i = 0; i < 7; i++) {
210cd1c6085SJohn Marino 			free((char *)tm.wday[i]);
211cd1c6085SJohn Marino 			tm.wday[i] = NULL;
212cd1c6085SJohn Marino 		}
213cd1c6085SJohn Marino 		break;
214cd1c6085SJohn Marino 	case T_DAY:
215cd1c6085SJohn Marino 		for (i = 0; i < 7; i++) {
216cd1c6085SJohn Marino 			free((char *)tm.weekday[i]);
217cd1c6085SJohn Marino 			tm.weekday[i] = NULL;
218cd1c6085SJohn Marino 		}
219cd1c6085SJohn Marino 		break;
220cd1c6085SJohn Marino 	case T_AM_PM:
221cd1c6085SJohn Marino 		free((char *)tm.am);
222cd1c6085SJohn Marino 		tm.am = NULL;
223cd1c6085SJohn Marino 		free((char *)tm.pm);
224cd1c6085SJohn Marino 		tm.pm = NULL;
225cd1c6085SJohn Marino 		break;
226cd1c6085SJohn Marino 	}
227cd1c6085SJohn Marino }
228cd1c6085SJohn Marino 
229cd1c6085SJohn Marino #pragma GCC diagnostic pop
230cd1c6085SJohn Marino 
231cd1c6085SJohn Marino void
dump_time(void)232cd1c6085SJohn Marino dump_time(void)
233cd1c6085SJohn Marino {
234cd1c6085SJohn Marino 	FILE *f;
235cd1c6085SJohn Marino 	int i;
236cd1c6085SJohn Marino 
237cd1c6085SJohn Marino 	if ((f = open_category()) == NULL) {
238cd1c6085SJohn Marino 		return;
239cd1c6085SJohn Marino 	}
240cd1c6085SJohn Marino 
241cd1c6085SJohn Marino 	for (i = 0; i < 12; i++) {
242cd1c6085SJohn Marino 		if (putl_category(tm.mon[i], f) == EOF) {
243cd1c6085SJohn Marino 			return;
244cd1c6085SJohn Marino 		}
245cd1c6085SJohn Marino 	}
246cd1c6085SJohn Marino 	for (i = 0; i < 12; i++) {
247cd1c6085SJohn Marino 		if (putl_category(tm.month[i], f) == EOF) {
248cd1c6085SJohn Marino 			return;
249cd1c6085SJohn Marino 		}
250cd1c6085SJohn Marino 	}
251cd1c6085SJohn Marino 	for (i = 0; i < 7; i++) {
252cd1c6085SJohn Marino 		if (putl_category(tm.wday[i], f) == EOF) {
253cd1c6085SJohn Marino 			return;
254cd1c6085SJohn Marino 		}
255cd1c6085SJohn Marino 	}
256cd1c6085SJohn Marino 	for (i = 0; i < 7; i++) {
257cd1c6085SJohn Marino 		if (putl_category(tm.weekday[i], f) == EOF) {
258cd1c6085SJohn Marino 			return;
259cd1c6085SJohn Marino 		}
260cd1c6085SJohn Marino 	}
261cd1c6085SJohn Marino 
262cd1c6085SJohn Marino 	/*
263cd1c6085SJohn Marino 	 * NOTE: If date_fmt is not specified, then we'll default to
264cd1c6085SJohn Marino 	 * using the %c for date.  This is reasonable for most
265cd1c6085SJohn Marino 	 * locales, although for reasons that I don't understand
266cd1c6085SJohn Marino 	 * Solaris historically has had a seperate format for date.
267cd1c6085SJohn Marino 	 */
268cd1c6085SJohn Marino 	if ((putl_category(tm.X_fmt, f) == EOF) ||
269cd1c6085SJohn Marino 	    (putl_category(tm.x_fmt, f) == EOF) ||
270cd1c6085SJohn Marino 	    (putl_category(tm.c_fmt, f) == EOF) ||
271cd1c6085SJohn Marino 	    (putl_category(tm.am, f) == EOF) ||
272cd1c6085SJohn Marino 	    (putl_category(tm.pm, f) == EOF) ||
273cd1c6085SJohn Marino 	    (putl_category(tm.date_fmt ? tm.date_fmt : tm.c_fmt, f) == EOF) ||
274cd1c6085SJohn Marino 	    (putl_category(tm.ampm_fmt, f) == EOF)) {
275cd1c6085SJohn Marino 		return;
276cd1c6085SJohn Marino 	}
277cd1c6085SJohn Marino 	close_category(f);
278cd1c6085SJohn Marino }
279