1 /* $NetBSD: localeio.c,v 1.5 2010/06/19 13:26:52 tnozaki Exp $ */
3 * Copyright (c) 2008, The NetBSD Foundation, Inc.
6 * This code is derived from software contributed to The NetBSD Foundation
9 * Redistribution and use in source and binary forms, with or without
10 * modification, are permitted provided that the following conditions
12 * 1. Redistributions of source code must retain the above copyright
13 * notice, this list of conditions and the following disclaimer.
14 * 2. Redistributions in binary form must reproduce the above copyright
15 * notice, this list of conditions and the following disclaimer in the
16 * documentation and/or other materials provided with the distribution.
18 * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
19 * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
20 * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
21 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
22 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
23 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
24 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
25 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
26 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
27 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
28 * POSSIBILITY OF SUCH DAMAGE.
31 #include <sys/cdefs.h>
32 #if defined(LIBC_SCCS) && !defined(lint)
33 __RCSID("$NetBSD: localeio.c,v 1.5 2010/06/19 13:26:52 tnozaki Exp $");
34 #endif /* LIBC_SCCS and not lint */
36 #include "namespace.h"
38 #include <sys/types.h>
55 _localeio_map_file(const char * __restrict path
,
56 void ** __restrict pvar
, size_t * __restrict plenvar
)
63 _DIAGASSERT(path
!= NULL
);
64 _DIAGASSERT(pvar
!= NULL
);
65 _DIAGASSERT(plenvar
!= NULL
);
67 fd
= open(path
, O_RDONLY
);
70 if (fcntl(fd
, F_SETFD
, FD_CLOEXEC
) == -1 || fstat(fd
, &st
) == 1) {
74 if (!S_ISREG(st
.st_mode
)) {
78 lenvar
= (size_t)st
.st_size
;
83 var
= mmap(NULL
, lenvar
, PROT_READ
,
84 MAP_FILE
|MAP_PRIVATE
, fd
, (off_t
)0);
85 if (var
== MAP_FAILED
) {
97 _localeio_unmap_file(void *var
, size_t lenvar
)
103 __loadlocale(const char *name
, size_t nstr
, size_t nbytes
,
104 size_t localesize
, void *currentlocale
)
107 unsigned char **ap
, *buf
, *bp
, *cp
, *cbp
, *ebp
;
108 unsigned char ***locale
;
112 _DIAGASSERT(name
!= NULL
);
113 _DIAGASSERT(localesize
!= 0);
114 _DIAGASSERT(currentlocale
!= NULL
);
116 if ((fd
= open(name
, O_RDONLY
)) == -1)
119 if ((fstat(fd
, &st
) == -1) || !S_ISREG(st
.st_mode
) ||
125 bufsize
= localesize
+ (size_t)st
.st_size
;
126 if ((buf
= malloc(bufsize
)) == NULL
) {
131 bp
= buf
+ localesize
;
132 if (read(fd
, bp
, (size_t)st
.st_size
) != st
.st_size
) {
137 ap
= (unsigned char **)(void *)buf
;
138 for (i
= (size_t)0, ebp
= buf
+ bufsize
; i
< nstr
; i
++) {
140 while (bp
!= ebp
&& *bp
!= '\n')
149 cp
= buf
+ (sizeof(unsigned char *) * nstr
);
150 for (i
= 0, cbp
= bp
; i
< nbytes
; i
++) {
153 while (bp
!= ebp
&& *bp
!= '\n')
159 /* ignore overflow/underflow and bad characters */
160 n
= (unsigned char)strtol((char *)cbp
, NULL
, 0);
161 cp
[i
] = (unsigned char)(n
& CHAR_MAX
);
165 locale
= currentlocale
;
167 *locale
= (unsigned char **)(void *)buf
;