1 1.5 tnozaki /* $NetBSD: citrus_lc_monetary.c,v 1.5 2012/03/04 21:14:55 tnozaki Exp $ */ 2 1.1 tnozaki 3 1.1 tnozaki /*- 4 1.1 tnozaki * Copyright (c)2008 Citrus Project, 5 1.1 tnozaki * All rights reserved. 6 1.1 tnozaki * 7 1.1 tnozaki * Redistribution and use in source and binary forms, with or without 8 1.1 tnozaki * modification, are permitted provided that the following conditions 9 1.1 tnozaki * are met: 10 1.1 tnozaki * 1. Redistributions of source code must retain the above copyright 11 1.1 tnozaki * notice, this list of conditions and the following disclaimer. 12 1.1 tnozaki * 2. Redistributions in binary form must reproduce the above copyright 13 1.1 tnozaki * notice, this list of conditions and the following disclaimer in the 14 1.1 tnozaki * documentation and/or other materials provided with the distribution. 15 1.1 tnozaki * 16 1.1 tnozaki * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 17 1.1 tnozaki * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 18 1.1 tnozaki * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 19 1.1 tnozaki * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 20 1.1 tnozaki * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 21 1.1 tnozaki * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 22 1.1 tnozaki * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 23 1.1 tnozaki * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 24 1.1 tnozaki * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 25 1.1 tnozaki * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 26 1.1 tnozaki * SUCH DAMAGE. 27 1.1 tnozaki */ 28 1.1 tnozaki 29 1.1 tnozaki #include <sys/cdefs.h> 30 1.1 tnozaki #if defined(LIBC_SCCS) && !defined(lint) 31 1.5 tnozaki __RCSID("$NetBSD: citrus_lc_monetary.c,v 1.5 2012/03/04 21:14:55 tnozaki Exp $"); 32 1.1 tnozaki #endif /* LIBC_SCCS and not lint */ 33 1.1 tnozaki 34 1.1 tnozaki #include "namespace.h" 35 1.1 tnozaki #include "reentrant.h" 36 1.1 tnozaki #include <sys/types.h> 37 1.1 tnozaki #include <sys/localedef.h> 38 1.1 tnozaki #include <sys/queue.h> 39 1.1 tnozaki #include <assert.h> 40 1.1 tnozaki #include <errno.h> 41 1.1 tnozaki #include <langinfo.h> 42 1.1 tnozaki #include <limits.h> 43 1.1 tnozaki #define __SETLOCALE_SOURCE__ 44 1.1 tnozaki #include <locale.h> 45 1.1 tnozaki #include <stddef.h> 46 1.1 tnozaki #include <stdio.h> 47 1.1 tnozaki #include <stdlib.h> 48 1.1 tnozaki #include <string.h> 49 1.1 tnozaki 50 1.1 tnozaki #include "setlocale_local.h" 51 1.1 tnozaki 52 1.1 tnozaki #include "citrus_namespace.h" 53 1.1 tnozaki #include "citrus_types.h" 54 1.1 tnozaki #include "citrus_bcs.h" 55 1.1 tnozaki #include "citrus_region.h" 56 1.1 tnozaki #include "citrus_lookup.h" 57 1.1 tnozaki #include "citrus_aliasname_local.h" 58 1.1 tnozaki #include "citrus_module.h" 59 1.1 tnozaki #include "citrus_mmap.h" 60 1.1 tnozaki #include "citrus_hash.h" 61 1.1 tnozaki #include "citrus_db.h" 62 1.1 tnozaki #include "citrus_db_hash.h" 63 1.1 tnozaki #include "citrus_memstream.h" 64 1.4 tnozaki #include "runetype_local.h" 65 1.1 tnozaki 66 1.1 tnozaki #include "fix_grouping.h" 67 1.1 tnozaki #include "citrus_fix_grouping.h" 68 1.1 tnozaki 69 1.1 tnozaki /* 70 1.1 tnozaki * macro required by all template headers 71 1.1 tnozaki */ 72 1.1 tnozaki #define _PREFIX(name) __CONCAT(_citrus_LC_MONETARY_, name) 73 1.1 tnozaki 74 1.1 tnozaki #include "nb_lc_monetary_misc.h" 75 1.1 tnozaki #include "citrus_lc_template_decl.h" 76 1.1 tnozaki 77 1.1 tnozaki static __inline void 78 1.1 tnozaki _citrus_LC_MONETARY_uninit(_MonetaryLocale *data) 79 1.1 tnozaki { 80 1.1 tnozaki _DIAGASSERT(data != NULL); 81 1.1 tnozaki 82 1.1 tnozaki free(__UNCONST(data->int_curr_symbol)); 83 1.1 tnozaki free(__UNCONST(data->currency_symbol)); 84 1.1 tnozaki free(__UNCONST(data->mon_decimal_point)); 85 1.1 tnozaki free(__UNCONST(data->mon_thousands_sep)); 86 1.1 tnozaki free(__UNCONST(data->mon_grouping)); 87 1.1 tnozaki free(__UNCONST(data->positive_sign)); 88 1.1 tnozaki free(__UNCONST(data->negative_sign)); 89 1.1 tnozaki } 90 1.1 tnozaki 91 1.1 tnozaki #include "citrus_lc_monetary.h" 92 1.1 tnozaki 93 1.1 tnozaki struct _citrus_LC_MONETARY_key { 94 1.1 tnozaki const char *name; 95 1.1 tnozaki size_t offset; 96 1.1 tnozaki }; 97 1.1 tnozaki 98 1.1 tnozaki #define OFFSET(field) (offsetof(_MonetaryLocale, field)) 99 1.1 tnozaki static const struct _citrus_LC_MONETARY_key keys_string[] = { 100 1.1 tnozaki { _CITRUS_LC_MONETARY_SYM_INT_CURR_SYMBOL, OFFSET(int_curr_symbol ) }, 101 1.1 tnozaki { _CITRUS_LC_MONETARY_SYM_CURRENCY_SYMBOL, OFFSET(currency_symbol ) }, 102 1.1 tnozaki { _CITRUS_LC_MONETARY_SYM_MON_DECIMAL_POINT, OFFSET(mon_decimal_point ) }, 103 1.1 tnozaki { _CITRUS_LC_MONETARY_SYM_MON_THOUSANDS_SEP, OFFSET(mon_thousands_sep ) }, 104 1.1 tnozaki { _CITRUS_LC_MONETARY_SYM_MON_GROUPING, OFFSET(mon_grouping ) }, 105 1.1 tnozaki { _CITRUS_LC_MONETARY_SYM_POSITIVE_SIGN, OFFSET(positive_sign ) }, 106 1.1 tnozaki { _CITRUS_LC_MONETARY_SYM_NEGATIVE_SIGN, OFFSET(negative_sign ) }, 107 1.1 tnozaki { NULL, (size_t)0 } 108 1.1 tnozaki }; 109 1.1 tnozaki static const struct _citrus_LC_MONETARY_key keys_char[] = { 110 1.1 tnozaki { _CITRUS_LC_MONETARY_SYM_INT_FRAC_DIGITS, OFFSET(int_frac_digits ) }, 111 1.1 tnozaki { _CITRUS_LC_MONETARY_SYM_FRAC_DIGITS, OFFSET(frac_digits ) }, 112 1.1 tnozaki { _CITRUS_LC_MONETARY_SYM_P_CS_PRECEDES, OFFSET(p_cs_precedes ) }, 113 1.1 tnozaki { _CITRUS_LC_MONETARY_SYM_P_SEP_BY_SPACE, OFFSET(p_sep_by_space ) }, 114 1.1 tnozaki { _CITRUS_LC_MONETARY_SYM_N_CS_PRECEDES, OFFSET(n_cs_precedes ) }, 115 1.1 tnozaki { _CITRUS_LC_MONETARY_SYM_N_SEP_BY_SPACE, OFFSET(n_sep_by_space ) }, 116 1.1 tnozaki { _CITRUS_LC_MONETARY_SYM_P_SIGN_POSN, OFFSET(p_sign_posn ) }, 117 1.1 tnozaki { _CITRUS_LC_MONETARY_SYM_N_SIGN_POSN, OFFSET(n_sign_posn ) }, 118 1.1 tnozaki { _CITRUS_LC_MONETARY_SYM_INT_P_CS_PRECEDES, OFFSET(int_p_cs_precedes ) }, 119 1.1 tnozaki { _CITRUS_LC_MONETARY_SYM_INT_N_CS_PRECEDES, OFFSET(int_n_cs_precedes ) }, 120 1.1 tnozaki { _CITRUS_LC_MONETARY_SYM_INT_P_SEP_BY_SPACE, OFFSET(int_p_sep_by_space) }, 121 1.1 tnozaki { _CITRUS_LC_MONETARY_SYM_INT_N_SEP_BY_SPACE, OFFSET(int_n_sep_by_space) }, 122 1.1 tnozaki { _CITRUS_LC_MONETARY_SYM_INT_P_SIGN_POSN, OFFSET(int_p_sign_posn ) }, 123 1.1 tnozaki { _CITRUS_LC_MONETARY_SYM_INT_N_SIGN_POSN, OFFSET(int_n_sign_posn ) }, 124 1.1 tnozaki { NULL, (size_t)0 } 125 1.1 tnozaki }; 126 1.1 tnozaki 127 1.1 tnozaki static __inline int 128 1.1 tnozaki _citrus_LC_MONETARY_init_normal(_MonetaryLocale * __restrict data, 129 1.1 tnozaki struct _citrus_db * __restrict db) 130 1.1 tnozaki { 131 1.1 tnozaki const struct _citrus_LC_MONETARY_key *key; 132 1.1 tnozaki char **p_string, *p_char; 133 1.1 tnozaki const char *s; 134 1.1 tnozaki uint8_t u8; 135 1.1 tnozaki 136 1.1 tnozaki _DIAGASSERT(data != NULL); 137 1.1 tnozaki _DIAGASSERT(db != NULL); 138 1.1 tnozaki 139 1.1 tnozaki memset(data, 0, sizeof(*data)); 140 1.1 tnozaki for (key = &keys_string[0]; key->name != NULL; ++key) { 141 1.1 tnozaki if (_db_lookupstr_by_s(db, key->name, &s, NULL)) 142 1.1 tnozaki goto fatal; 143 1.1 tnozaki p_string = (char **)(void *) 144 1.1 tnozaki (((char *)(void *)data) + key->offset); 145 1.1 tnozaki *p_string = strdup(s); 146 1.1 tnozaki if (*p_string == NULL) 147 1.1 tnozaki goto fatal; 148 1.1 tnozaki } 149 1.1 tnozaki for (key = &keys_char[0]; key->name != NULL; ++key) { 150 1.1 tnozaki if (_db_lookup8_by_s(db, key->name, &u8, NULL)) 151 1.1 tnozaki goto fatal; 152 1.1 tnozaki p_char = ((char *)(void *)data) + key->offset; 153 1.1 tnozaki *p_char = (char)(unsigned char)u8; 154 1.1 tnozaki } 155 1.1 tnozaki _CITRUS_FIXUP_CHAR_MAX_MD(data->mon_grouping); 156 1.1 tnozaki 157 1.1 tnozaki return 0; 158 1.1 tnozaki 159 1.1 tnozaki fatal: 160 1.1 tnozaki _citrus_LC_MONETARY_uninit(data); 161 1.1 tnozaki return EFTYPE; 162 1.1 tnozaki } 163 1.1 tnozaki 164 1.1 tnozaki static __inline int 165 1.1 tnozaki _citrus_LC_MONETARY_init_fallback(_MonetaryLocale * __restrict data, 166 1.1 tnozaki struct _memstream * __restrict ms) 167 1.1 tnozaki { 168 1.1 tnozaki const struct _citrus_LC_MONETARY_key *key; 169 1.1 tnozaki char **p_string, *p_char; 170 1.1 tnozaki const char *s; 171 1.1 tnozaki size_t n; 172 1.1 tnozaki char *t; 173 1.1 tnozaki long int l; 174 1.1 tnozaki 175 1.1 tnozaki _DIAGASSERT(data != NULL); 176 1.1 tnozaki _DIAGASSERT(ms != NULL); 177 1.1 tnozaki 178 1.1 tnozaki memset(data, 0, sizeof(*data)); 179 1.1 tnozaki for (key = &keys_string[0]; key->name != NULL; ++key) { 180 1.1 tnozaki if ((s = _memstream_getln(ms, &n)) == NULL) 181 1.1 tnozaki goto fatal; 182 1.1 tnozaki p_string = (char **)(void *) 183 1.1 tnozaki (((char *)(void *)data) + key->offset); 184 1.1 tnozaki *p_string = strndup(s, n - 1); 185 1.1 tnozaki if (*p_string == NULL) 186 1.1 tnozaki goto fatal; 187 1.1 tnozaki } 188 1.1 tnozaki for (key = &keys_char[0]; key->name != NULL; ++key) { 189 1.1 tnozaki if ((s = _memstream_getln(ms, &n)) == NULL) 190 1.1 tnozaki goto fatal; 191 1.1 tnozaki t = strndup(s, n - 1); 192 1.1 tnozaki if (t == NULL) 193 1.1 tnozaki goto fatal; 194 1.1 tnozaki s = (const char *)t; 195 1.1 tnozaki l = _bcs_strtol(s, &t, 0); 196 1.1 tnozaki if (s == t || l < 0 || l > 0x7fL) { 197 1.1 tnozaki free(t); 198 1.1 tnozaki goto fatal; 199 1.1 tnozaki } 200 1.1 tnozaki free(t); 201 1.1 tnozaki p_char = ((char *)(void *)data) + key->offset; 202 1.1 tnozaki *p_char = (char)(l & 0x7fL); 203 1.1 tnozaki } 204 1.1 tnozaki data->mon_grouping = 205 1.1 tnozaki __fix_locale_grouping_str(data->mon_grouping); 206 1.1 tnozaki return 0; 207 1.1 tnozaki 208 1.1 tnozaki fatal: 209 1.1 tnozaki _citrus_LC_MONETARY_uninit(data); 210 1.1 tnozaki return EFTYPE; 211 1.1 tnozaki } 212 1.1 tnozaki 213 1.1 tnozaki /* 214 1.1 tnozaki * macro required by citrus_lc_template.h 215 1.1 tnozaki */ 216 1.1 tnozaki #define _CATEGORY_DB "LC_MONETARY" 217 1.1 tnozaki #define _CATEGORY_MAGIC _CITRUS_LC_MONETARY_MAGIC_1 218 1.1 tnozaki 219 1.1 tnozaki #include "citrus_lc_template.h" 220