Expand PMF_FN_* macros.
[netbsd-mini2440.git] / lib / libc / locale / generic_lc_all.c
blobd90da9e540e5db6fa3e7e91e29b6b215b758129e
1 /* $NetBSD: generic_lc_all.c,v 1.2 2009/01/11 02:46:28 christos Exp $ */
3 /*-
4 * Copyright (c)2008 Citrus Project,
5 * All rights reserved.
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions
9 * are met:
10 * 1. Redistributions of source code must retain the above copyright
11 * notice, this list of conditions and the following disclaimer.
12 * 2. Redistributions in binary form must reproduce the above copyright
13 * notice, this list of conditions and the following disclaimer in the
14 * documentation and/or other materials provided with the distribution.
16 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
17 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
20 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
21 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
22 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
23 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
24 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
25 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
26 * SUCH DAMAGE.
29 #include <sys/cdefs.h>
30 #if defined(LIBC_SCCS) && !defined(lint)
31 __RCSID("$NetBSD: generic_lc_all.c,v 1.2 2009/01/11 02:46:28 christos Exp $");
32 #endif /* LIBC_SCCS and not lint */
34 #include <sys/types.h>
35 #include <assert.h>
36 #include <langinfo.h>
37 #define __SETLOCALE_SOURCE__
38 #include <locale.h>
39 #include <stdio.h>
40 #include <stdlib.h>
41 #include <string.h>
43 #include "setlocale_local.h"
46 * macro required by all template headers
48 #define _PREFIX(name) __CONCAT(_generic_LC_ALL_, name)
50 #include "generic_lc_template_decl.h"
52 const char *
53 _generic_LC_ALL_setlocale(const char * __restrict name,
54 struct _locale_impl_t * __restrict locale)
56 _locale_category_t *l;
57 char head[_LOCALENAME_LEN_MAX * (_LC_LAST - 1)], *tail;
58 const char *tokens[_LC_LAST], *s, *t;
59 int load_locale_success, i, j;
61 l = _find_category(1);
62 _DIAGASSERT(l != NULL);
63 load_locale_success = 0;
64 if (name != NULL) {
65 strlcpy(&head[0], name, sizeof(head));
66 tokens[1] = &head[0];
67 tail = strchr(tokens[1], '/');
68 if (tail == NULL) {
69 for (i = 2; i < _LC_LAST; ++i)
70 tokens[i] = tokens[1];
71 } else {
72 *tail++ = '\0';
73 for (i = 2; i < _LC_LAST - 1; ++i) {
74 tokens[i] = (const char *)tail;
75 tail = strchr(tokens[i], '/');
76 if (tail == NULL)
77 return NULL;
78 *tail++ = '\0';
80 tokens[_LC_LAST - 1] = (const char *)tail;
81 tail = strchr(tokens[i], '/');
82 if (tail != NULL)
83 return NULL;
85 if ((*l->setlocale)(tokens[1], locale) != NULL)
86 load_locale_success = 1;
88 s = (*l->setlocale)(NULL, locale);
89 _DIAGASSERT(s != NULL);
90 strlcpy(&locale->query[0], s, sizeof(locale->query));
91 for (i = 2, j = 0; i < _LC_LAST; ++i) {
92 l = _find_category(i);
93 _DIAGASSERT(l != NULL);
94 if (name != NULL) {
95 if ((*l->setlocale)(tokens[i], locale) != NULL)
96 load_locale_success = 1;
98 t = (*l->setlocale)(NULL, locale);
99 _DIAGASSERT(t != NULL);
100 if (j == 0) {
101 if (!strcmp(s, t))
102 continue;
103 for (j = 2; j < i; ++j) {
104 strlcat(&locale->query[0], "/",
105 sizeof(locale->query));
106 strlcat(&locale->query[0], s,
107 sizeof(locale->query));
110 strlcat(&locale->query[0], "/", sizeof(locale->query));
111 strlcat(&locale->query[0], t, sizeof(locale->query));
113 if (name != NULL && !load_locale_success)
114 return NULL;
115 return (const char *)&locale->query[0];
119 * macro requrired by generic_lc_template.h
121 #define _CATEGORY_ID LC_ALL
123 #include "generic_lc_template.h"
124 _LOCALE_CATEGORY_ENTRY(_generic_LC_ALL_);