]> git.saurik.com Git - apple/libc.git/blame - locale/localeconv-fbsd.c
Libc-498.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
49/*------------------------------------------------------------------------
50 * PR-3417676: We need to provide a way to force "C" locale style number
51 * formatting independent of the locale setting. We provide private
52 * routines to get and set a flag that tells localeconv() to either return
53 * a "C" struct lconv, or the one dependent on the actual locale.
54 *------------------------------------------------------------------------*/
55static char empty[] = "";
56static char numempty[] = { CHAR_MAX, '\0' };
57
58/*
59 * Default (C) locale conversion.
60 */
61static struct lconv _C_lconv = {
62 ".", /* decimal_point */
63 empty, /* thousands_sep */
64 numempty, /* grouping */
65 empty, /* int_curr_symbol */
66 empty, /* currency_symbol */
67 empty, /* mon_decimal_point */
68 empty, /* mon_thousands_sep */
69 numempty, /* mon_grouping */
70 empty, /* positive_sign */
71 empty, /* negative_sign */
72 CHAR_MAX, /* int_frac_digits */
73 CHAR_MAX, /* frac_digits */
74 CHAR_MAX, /* p_cs_precedes */
75 CHAR_MAX, /* p_sep_by_space */
76 CHAR_MAX, /* n_cs_precedes */
77 CHAR_MAX, /* n_sep_by_space */
78 CHAR_MAX, /* p_sign_posn */
79 CHAR_MAX, /* n_sign_posn */
80 CHAR_MAX, /* int_p_cs_precedes */
81 CHAR_MAX, /* int_n_cs_precedes */
82 CHAR_MAX, /* int_p_sep_by_space */
83 CHAR_MAX, /* int_n_sep_by_space */
84 CHAR_MAX, /* int_p_sign_posn */
85 CHAR_MAX, /* int_n_sign_posn */
86};
87static int _onlyClocaleconv = 0;
88
89int
90__getonlyClocaleconv(void)
91{
92 return _onlyClocaleconv;
93}
94
95int
96__setonlyClocaleconv(int val)
97{
98 int prev = _onlyClocaleconv;
99
100 _onlyClocaleconv = val;
101 return prev;
102}
103
104/*
105 * The localeconv() function constructs a struct lconv from the current
106 * monetary and numeric locales.
107 *
108 * Because localeconv() may be called many times (especially by library
109 * routines like printf() & strtod()), the approprate members of the
110 * lconv structure are computed only when the monetary or numeric
111 * locale has been changed.
112 */
113
114/*
115 * Return the current locale conversion.
116 */
117struct lconv *
118localeconv_l(locale_t loc)
119{
120 struct __xlocale_st_localeconv *lc;
121
122 NORMALIZE_LOCALE(loc);
123 if (loc->__lc_localeconv && !loc->__mlocale_changed && !loc->__nlocale_changed)
124 return &loc->__lc_localeconv->__ret;
125
126 lc = (struct __xlocale_st_localeconv *)malloc(sizeof(struct __xlocale_st_localeconv));
127 lc->__refcount = 1;
128 lc->__free_extra = NULL;
129 if (loc->__lc_localeconv)
130 lc->__ret = loc->__lc_localeconv->__ret;
131 else {
132 loc->__mlocale_changed = 1;
133 loc->__nlocale_changed = 1;
134 }
135
136 if (loc->__mlocale_changed) {
137 /* LC_MONETARY part */
138 struct lc_monetary_T * mptr;
139
140#define M_ASSIGN_STR(NAME) (lc->__ret.NAME = (char*)mptr->NAME)
141#define M_ASSIGN_CHAR(NAME) (lc->__ret.NAME = mptr->NAME[0])
142
143 mptr = __get_current_monetary_locale(loc);
144 M_ASSIGN_STR(int_curr_symbol);
145 M_ASSIGN_STR(currency_symbol);
146 M_ASSIGN_STR(mon_decimal_point);
147 M_ASSIGN_STR(mon_thousands_sep);
148 M_ASSIGN_STR(mon_grouping);
149 M_ASSIGN_STR(positive_sign);
150 M_ASSIGN_STR(negative_sign);
151 M_ASSIGN_CHAR(int_frac_digits);
152 M_ASSIGN_CHAR(frac_digits);
153 M_ASSIGN_CHAR(p_cs_precedes);
154 M_ASSIGN_CHAR(p_sep_by_space);
155 M_ASSIGN_CHAR(n_cs_precedes);
156 M_ASSIGN_CHAR(n_sep_by_space);
157 M_ASSIGN_CHAR(p_sign_posn);
158 M_ASSIGN_CHAR(n_sign_posn);
159 M_ASSIGN_CHAR(int_p_cs_precedes);
160 M_ASSIGN_CHAR(int_n_cs_precedes);
161 M_ASSIGN_CHAR(int_p_sep_by_space);
162 M_ASSIGN_CHAR(int_n_sep_by_space);
163 M_ASSIGN_CHAR(int_p_sign_posn);
164 M_ASSIGN_CHAR(int_n_sign_posn);
165 loc->__mlocale_changed = 0;
166 }
167
168 if (loc->__nlocale_changed) {
169 /* LC_NUMERIC part */
170 struct lc_numeric_T * nptr;
171
172#define N_ASSIGN_STR(NAME) (lc->__ret.NAME = (char*)nptr->NAME)
173
174 nptr = __get_current_numeric_locale(loc);
175 N_ASSIGN_STR(decimal_point);
176 N_ASSIGN_STR(thousands_sep);
177 N_ASSIGN_STR(grouping);
178 loc->__nlocale_changed = 0;
179 }
180
181 XL_RELEASE(loc->__lc_localeconv);
182 loc->__lc_localeconv = lc;
183
184 return (&lc->__ret);
185}
186
187/*
188 * Return the current locale conversion.
189 */
190struct lconv *
191localeconv()
192{
193 /*--------------------------------------------------------------------
194 * If _onlyClocaleconv is non-zero, just return __lconv, which is a "C"
195 * struct lconv *. Otherwise, do the normal thing.
196 *--------------------------------------------------------------------*/
197 if (_onlyClocaleconv)
198 return &_C_lconv;
199 return localeconv_l(__current_locale());
200}