textdomain.c revision 1.12 1 1.12 junyoung /* $NetBSD: textdomain.c,v 1.12 2007/09/25 08:19:09 junyoung Exp $ */
2 1.1 itojun
3 1.1 itojun /*-
4 1.5 minoura * Copyright (c) 2000, 2001 Citrus Project,
5 1.1 itojun * All rights reserved.
6 1.1 itojun *
7 1.1 itojun * Redistribution and use in source and binary forms, with or without
8 1.1 itojun * modification, are permitted provided that the following conditions
9 1.1 itojun * are met:
10 1.1 itojun * 1. Redistributions of source code must retain the above copyright
11 1.1 itojun * notice, this list of conditions and the following disclaimer.
12 1.1 itojun * 2. Redistributions in binary form must reproduce the above copyright
13 1.1 itojun * notice, this list of conditions and the following disclaimer in the
14 1.1 itojun * documentation and/or other materials provided with the distribution.
15 1.1 itojun *
16 1.1 itojun * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
17 1.1 itojun * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18 1.1 itojun * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19 1.1 itojun * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
20 1.1 itojun * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
21 1.1 itojun * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
22 1.1 itojun * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
23 1.1 itojun * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
24 1.1 itojun * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
25 1.1 itojun * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
26 1.1 itojun * SUCH DAMAGE.
27 1.1 itojun */
28 1.1 itojun
29 1.1 itojun #include <sys/cdefs.h>
30 1.12 junyoung __RCSID("$NetBSD: textdomain.c,v 1.12 2007/09/25 08:19:09 junyoung Exp $");
31 1.1 itojun
32 1.1 itojun #include <sys/param.h>
33 1.1 itojun
34 1.1 itojun #include <stdio.h>
35 1.1 itojun #include <string.h>
36 1.4 itojun #include <stdlib.h>
37 1.1 itojun #include <libintl.h>
38 1.1 itojun #include "libintl_local.h"
39 1.1 itojun #include "pathnames.h"
40 1.1 itojun
41 1.5 minoura static struct domainbinding __default_binding = {
42 1.5 minoura NULL, DEFAULT_DOMAINNAME, _PATH_TEXTDOMAIN,
43 1.5 minoura };
44 1.5 minoura struct domainbinding *__bindings = &__default_binding;
45 1.5 minoura char __current_domainname[PATH_MAX] = DEFAULT_DOMAINNAME;
46 1.1 itojun
47 1.12 junyoung static struct domainbinding *domainbinding_lookup(const char *, int);
48 1.10 yamt
49 1.1 itojun /*
50 1.1 itojun * set the default domainname for dcngettext() and friends.
51 1.1 itojun */
52 1.1 itojun char *
53 1.12 junyoung textdomain(const char *domainname)
54 1.1 itojun {
55 1.1 itojun
56 1.2 itojun /* NULL pointer gives the current setting */
57 1.4 itojun if (!domainname)
58 1.5 minoura return __current_domainname;
59 1.2 itojun
60 1.2 itojun /* empty string sets the value back to the default */
61 1.4 itojun if (!*domainname) {
62 1.5 minoura strlcpy(__current_domainname, DEFAULT_DOMAINNAME,
63 1.5 minoura sizeof(__current_domainname));
64 1.4 itojun } else {
65 1.5 minoura strlcpy(__current_domainname, domainname,
66 1.5 minoura sizeof(__current_domainname));
67 1.4 itojun }
68 1.5 minoura return __current_domainname;
69 1.1 itojun }
70 1.1 itojun
71 1.1 itojun char *
72 1.12 junyoung bindtextdomain(const char *domainname, const char *dirname)
73 1.1 itojun {
74 1.4 itojun struct domainbinding *p;
75 1.2 itojun
76 1.2 itojun /* NULL pointer or empty string returns NULL with no operation */
77 1.2 itojun if (!domainname || !*domainname)
78 1.2 itojun return NULL;
79 1.1 itojun
80 1.8 drochner if (dirname && (strlen(dirname) + 1 > sizeof(p->path)))
81 1.1 itojun return NULL;
82 1.6 yamt
83 1.6 yamt #if 0
84 1.1 itojun /* disallow relative path */
85 1.1 itojun if (dirname[0] != '/')
86 1.1 itojun return NULL;
87 1.6 yamt #endif
88 1.1 itojun
89 1.4 itojun if (strlen(domainname) + 1 > sizeof(p->domainname))
90 1.1 itojun return NULL;
91 1.1 itojun
92 1.10 yamt p = domainbinding_lookup(domainname, (dirname != NULL));
93 1.8 drochner
94 1.8 drochner if (!dirname) {
95 1.8 drochner if (p)
96 1.8 drochner return (p->path);
97 1.8 drochner else
98 1.11 tshiozak return (char *)__UNCONST(_PATH_TEXTDOMAIN);
99 1.8 drochner }
100 1.8 drochner
101 1.4 itojun strlcpy(p->path, dirname, sizeof(p->path));
102 1.5 minoura p->mohandle.mo.mo_magic = 0; /* invalidate current mapping */
103 1.1 itojun
104 1.8 drochner return (p->path);
105 1.1 itojun }
106 1.1 itojun
107 1.1 itojun char *
108 1.12 junyoung bind_textdomain_codeset(const char *domainname, const char *codeset)
109 1.1 itojun {
110 1.10 yamt struct domainbinding *p;
111 1.10 yamt
112 1.10 yamt p = domainbinding_lookup(domainname, (codeset != NULL));
113 1.10 yamt if (p == NULL)
114 1.10 yamt return NULL;
115 1.10 yamt
116 1.10 yamt if (codeset) {
117 1.10 yamt if (p->codeset)
118 1.10 yamt free(p->codeset);
119 1.10 yamt p->codeset = strdup(codeset);
120 1.10 yamt }
121 1.10 yamt
122 1.10 yamt return p->codeset;
123 1.10 yamt }
124 1.10 yamt
125 1.10 yamt /*
126 1.10 yamt * lookup binding for the domainname
127 1.10 yamt */
128 1.10 yamt static struct domainbinding *
129 1.12 junyoung domainbinding_lookup(const char *domainname, int alloc)
130 1.10 yamt {
131 1.10 yamt struct domainbinding *p;
132 1.10 yamt
133 1.10 yamt for (p = __bindings; p; p = p->next)
134 1.10 yamt if (strcmp(p->domainname, domainname) == 0)
135 1.10 yamt break;
136 1.10 yamt
137 1.10 yamt if (!p && alloc) {
138 1.10 yamt p = (struct domainbinding *)malloc(sizeof(*p));
139 1.10 yamt if (!p)
140 1.10 yamt return NULL;
141 1.10 yamt memset(p, 0, sizeof(*p));
142 1.10 yamt p->next = __bindings;
143 1.10 yamt strlcpy(p->domainname, domainname, sizeof(p->domainname));
144 1.10 yamt __bindings = p;
145 1.10 yamt }
146 1.1 itojun
147 1.10 yamt return p;
148 1.1 itojun }
149