]> git.saurik.com Git - apple/libc.git/blame - locale/localeconv-fbsd.c
Libc-498.1.7.tar.gz
[apple/libc.git] / locale / localeconv-fbsd.c
CommitLineData
224c7076
A
1/*
2 * Copyright (c) 2001 Alexey Zelkin <phantom@FreeBSD.org>
3 * Copyright (c) 1991, 1993
4 * The Regents of the University of California. All rights reserved.
5 *
6 * Redistribution and use in source and binary forms, with or without
7 * modification, are permitted provided that the following conditions
8 * are met:
9 * 1. Redistributions of source code must retain the above copyright
10 * notice, this list of conditions and the following disclaimer.
11 * 2. Redistributions in binary form must reproduce the above copyright
12 * notice, this list of conditions and the following disclaimer in the
13 * documentation and/or other materials provided with the distribution.
14 * 3. All advertising materials mentioning features or use of this software
15 * must display the following acknowledgement:
16 * This product includes software developed by the University of
17 * California, Berkeley and its contributors.
18 * 4. Neither the name of the University nor the names of its contributors
19 * may be used to endorse or promote products derived from this software
20 * without specific prior written permission.
21 *
22 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
23 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
24 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
25 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
26 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
27 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
28 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
29 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
30 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
31 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
32 * SUCH DAMAGE.
33 */
34
35#if defined(LIBC_SCCS) && !defined(lint)
36static char sccsid[] = "@(#)localeconv.c 8.1 (Berkeley) 6/4/93";
37#endif /* LIBC_SCCS and not lint */
38#include <sys/cdefs.h>
39__FBSDID("$FreeBSD: src/lib/libc/locale/localeconv.c,v 1.13 2003/06/26 10:46:16 phantom Exp $");
40
41#include "xlocale_private.h"
42
43#include <limits.h>
44#include <locale.h>
45
46#include "lmonetary.h"
47#include "lnumeric.h"
48
b5d655f7 49#ifdef __APPLE_PR3417676_HACK__
224c7076
A
50/*------------------------------------------------------------------------
51 * PR-3417676: We need to provide a way to force "C" locale style number
52 * formatting independent of the locale setting. We provide private
53 * routines to get and set a flag that tells localeconv() to either return
54 * a "C" struct lconv, or the one dependent on the actual locale.
55 *------------------------------------------------------------------------*/
56static char empty[] = "";
57static char numempty[] = { CHAR_MAX, '\0' };
58
59/*
60 * Default (C) locale conversion.
61 */
62static struct lconv _C_lconv = {
63 ".", /* decimal_point */
64 empty, /* thousands_sep */
65 numempty, /* grouping */
66 empty, /* int_curr_symbol */
67 empty, /* currency_symbol */
68 empty, /* mon_decimal_point */
69 empty, /* mon_thousands_sep */
70 numempty, /* mon_grouping */
71 empty, /* positive_sign */
72 empty, /* negative_sign */
73 CHAR_MAX, /* int_frac_digits */
74 CHAR_MAX, /* frac_digits */
75 CHAR_MAX, /* p_cs_precedes */
76 CHAR_MAX, /* p_sep_by_space */
77 CHAR_MAX, /* n_cs_precedes */
78 CHAR_MAX, /* n_sep_by_space */
79 CHAR_MAX, /* p_sign_posn */
80 CHAR_MAX, /* n_sign_posn */
81 CHAR_MAX, /* int_p_cs_precedes */
82 CHAR_MAX, /* int_n_cs_precedes */
83 CHAR_MAX, /* int_p_sep_by_space */
84 CHAR_MAX, /* int_n_sep_by_space */
85 CHAR_MAX, /* int_p_sign_posn */
86 CHAR_MAX, /* int_n_sign_posn */
87};
88static int _onlyClocaleconv = 0;
89
90int
91__getonlyClocaleconv(void)
92{
93 return _onlyClocaleconv;
94}
95
96int
97__setonlyClocaleconv(int val)
98{
99 int prev = _onlyClocaleconv;
100
101 _onlyClocaleconv = val;
102 return prev;
103}
b5d655f7 104#endif /* __APPLE_PR3417676_HACK__ */
224c7076
A
105
106/*
107 * The localeconv() function constructs a struct lconv from the current
108 * monetary and numeric locales.
109 *
110 * Because localeconv() may be called many times (especially by library
111 * routines like printf() & strtod()), the approprate members of the
112 * lconv structure are computed only when the monetary or numeric
113 * locale has been changed.
114 */
115
116/*
117 * Return the current locale conversion.
118 */
119struct lconv *
120localeconv_l(locale_t loc)
121{
122 struct __xlocale_st_localeconv *lc;
123
124 NORMALIZE_LOCALE(loc);
125 if (loc->__lc_localeconv && !loc->__mlocale_changed && !loc->__nlocale_changed)
126 return &loc->__lc_localeconv->__ret;
127
128 lc = (struct __xlocale_st_localeconv *)malloc(sizeof(struct __xlocale_st_localeconv));
129 lc->__refcount = 1;
130 lc->__free_extra = NULL;
131 if (loc->__lc_localeconv)
132 lc->__ret = loc->__lc_localeconv->__ret;
133 else {
134 loc->__mlocale_changed = 1;
135 loc->__nlocale_changed = 1;
136 }
137
138 if (loc->__mlocale_changed) {
139 /* LC_MONETARY part */
140 struct lc_monetary_T * mptr;
141
142#define M_ASSIGN_STR(NAME) (lc->__ret.NAME = (char*)mptr->NAME)
143#define M_ASSIGN_CHAR(NAME) (lc->__ret.NAME = mptr->NAME[0])
144
145 mptr = __get_current_monetary_locale(loc);
146 M_ASSIGN_STR(int_curr_symbol);
147 M_ASSIGN_STR(currency_symbol);
148 M_ASSIGN_STR(mon_decimal_point);
149 M_ASSIGN_STR(mon_thousands_sep);
150 M_ASSIGN_STR(mon_grouping);
151 M_ASSIGN_STR(positive_sign);
152 M_ASSIGN_STR(negative_sign);
153 M_ASSIGN_CHAR(int_frac_digits);
154 M_ASSIGN_CHAR(frac_digits);
155 M_ASSIGN_CHAR(p_cs_precedes);
156 M_ASSIGN_CHAR(p_sep_by_space);
157 M_ASSIGN_CHAR(n_cs_precedes);
158 M_ASSIGN_CHAR(n_sep_by_space);
159 M_ASSIGN_CHAR(p_sign_posn);
160 M_ASSIGN_CHAR(n_sign_posn);
161 M_ASSIGN_CHAR(int_p_cs_precedes);
162 M_ASSIGN_CHAR(int_n_cs_precedes);
163 M_ASSIGN_CHAR(int_p_sep_by_space);
164 M_ASSIGN_CHAR(int_n_sep_by_space);
165 M_ASSIGN_CHAR(int_p_sign_posn);
166 M_ASSIGN_CHAR(int_n_sign_posn);
167 loc->__mlocale_changed = 0;
168 }
169
170 if (loc->__nlocale_changed) {
171 /* LC_NUMERIC part */
172 struct lc_numeric_T * nptr;
173
174#define N_ASSIGN_STR(NAME) (lc->__ret.NAME = (char*)nptr->NAME)
175
176 nptr = __get_current_numeric_locale(loc);
177 N_ASSIGN_STR(decimal_point);
178 N_ASSIGN_STR(thousands_sep);
179 N_ASSIGN_STR(grouping);
180 loc->__nlocale_changed = 0;
181 }
182
183 XL_RELEASE(loc->__lc_localeconv);
184 loc->__lc_localeconv = lc;
185
186 return (&lc->__ret);
187}
188
189/*
190 * Return the current locale conversion.
191 */
192struct lconv *
193localeconv()
194{
b5d655f7 195#ifdef __APPLE_PR3417676_HACK__
224c7076
A
196 /*--------------------------------------------------------------------
197 * If _onlyClocaleconv is non-zero, just return __lconv, which is a "C"
198 * struct lconv *. Otherwise, do the normal thing.
199 *--------------------------------------------------------------------*/
200 if (_onlyClocaleconv)
201 return &_C_lconv;
b5d655f7 202#endif /* __APPLE_PR3417676_HACK__ */
224c7076
A
203 return localeconv_l(__current_locale());
204}