cygwin: add catopen, catgets, catclose
The implementation is taken from FreeBSD with #ifdef __CYGWIN__ modifications. Signed-off-by: Yaakov Selkowitz <yselkowi@redhat.com>
This commit is contained in:
parent
bd62f539de
commit
cadc12f695
@ -333,6 +333,7 @@ DLL_OFILES:= \
|
|||||||
mktemp.o \
|
mktemp.o \
|
||||||
mmap.o \
|
mmap.o \
|
||||||
msg.o \
|
msg.o \
|
||||||
|
msgcat.o \
|
||||||
mount.o \
|
mount.o \
|
||||||
net.o \
|
net.o \
|
||||||
netdb.o \
|
netdb.o \
|
||||||
|
@ -274,6 +274,9 @@ catanh NOSIGFE
|
|||||||
catanhf NOSIGFE
|
catanhf NOSIGFE
|
||||||
catanhl NOSIGFE
|
catanhl NOSIGFE
|
||||||
catanl NOSIGFE
|
catanl NOSIGFE
|
||||||
|
catclose SIGFE
|
||||||
|
catgets SIGFE
|
||||||
|
catopen SIGFE
|
||||||
cbrt NOSIGFE
|
cbrt NOSIGFE
|
||||||
cbrtf NOSIGFE
|
cbrtf NOSIGFE
|
||||||
cbrtl NOSIGFE
|
cbrtl NOSIGFE
|
||||||
|
@ -493,12 +493,13 @@ details. */
|
|||||||
322: [w]scanf %m modifier.
|
322: [w]scanf %m modifier.
|
||||||
323: scanf %l[ conversion.
|
323: scanf %l[ conversion.
|
||||||
324: Export sigtimedwait.
|
324: Export sigtimedwait.
|
||||||
|
325: Export catclose, catgets, catopen.
|
||||||
|
|
||||||
Note that we forgot to bump the api for ualarm, strtoll, strtoull,
|
Note that we forgot to bump the api for ualarm, strtoll, strtoull,
|
||||||
sigaltstack, sethostname. */
|
sigaltstack, sethostname. */
|
||||||
|
|
||||||
#define CYGWIN_VERSION_API_MAJOR 0
|
#define CYGWIN_VERSION_API_MAJOR 0
|
||||||
#define CYGWIN_VERSION_API_MINOR 324
|
#define CYGWIN_VERSION_API_MINOR 325
|
||||||
|
|
||||||
/* There is also a compatibity version number associated with the shared memory
|
/* There is also a compatibity version number associated with the shared memory
|
||||||
regions. It is incremented when incompatible changes are made to the shared
|
regions. It is incremented when incompatible changes are made to the shared
|
||||||
|
100
winsup/cygwin/include/nl_types.h
Normal file
100
winsup/cygwin/include/nl_types.h
Normal file
@ -0,0 +1,100 @@
|
|||||||
|
/* $NetBSD: nl_types.h,v 1.9 2000/10/03 19:53:32 sommerfeld Exp $ */
|
||||||
|
|
||||||
|
/*-
|
||||||
|
* SPDX-License-Identifier: BSD-2-Clause-NetBSD
|
||||||
|
*
|
||||||
|
* Copyright (c) 1996 The NetBSD Foundation, Inc.
|
||||||
|
* All rights reserved.
|
||||||
|
*
|
||||||
|
* This code is derived from software contributed to The NetBSD Foundation
|
||||||
|
* by J.T. Conklin.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions
|
||||||
|
* are met:
|
||||||
|
* 1. Redistributions of source code must retain the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer.
|
||||||
|
* 2. Redistributions in binary form must reproduce the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer in the
|
||||||
|
* documentation and/or other materials provided with the distribution.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
|
||||||
|
* ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
|
||||||
|
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
||||||
|
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
|
||||||
|
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
||||||
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
||||||
|
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
||||||
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
||||||
|
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
||||||
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
||||||
|
* POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*
|
||||||
|
* $FreeBSD$
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef _NL_TYPES_H_
|
||||||
|
#define _NL_TYPES_H_
|
||||||
|
|
||||||
|
#include <sys/cdefs.h>
|
||||||
|
#include <sys/types.h>
|
||||||
|
|
||||||
|
#ifdef _NLS_PRIVATE
|
||||||
|
/*
|
||||||
|
* MESSAGE CATALOG FILE FORMAT.
|
||||||
|
*
|
||||||
|
* The NetBSD/FreeBSD message catalog format is similar to the format used by
|
||||||
|
* Svr4 systems. The differences are:
|
||||||
|
* * fixed byte order (big endian)
|
||||||
|
* * fixed data field sizes
|
||||||
|
*
|
||||||
|
* A message catalog contains four data types: a catalog header, one
|
||||||
|
* or more set headers, one or more message headers, and one or more
|
||||||
|
* text strings.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#define _NLS_MAGIC 0xff88ff89
|
||||||
|
|
||||||
|
struct _nls_cat_hdr {
|
||||||
|
int32_t __magic;
|
||||||
|
int32_t __nsets;
|
||||||
|
int32_t __mem;
|
||||||
|
int32_t __msg_hdr_offset;
|
||||||
|
int32_t __msg_txt_offset;
|
||||||
|
} ;
|
||||||
|
|
||||||
|
struct _nls_set_hdr {
|
||||||
|
int32_t __setno; /* set number: 0 < x <= NL_SETMAX */
|
||||||
|
int32_t __nmsgs; /* number of messages in the set */
|
||||||
|
int32_t __index; /* index of first msg_hdr in msg_hdr table */
|
||||||
|
} ;
|
||||||
|
|
||||||
|
struct _nls_msg_hdr {
|
||||||
|
int32_t __msgno; /* msg number: 0 < x <= NL_MSGMAX */
|
||||||
|
int32_t __msglen;
|
||||||
|
int32_t __offset;
|
||||||
|
} ;
|
||||||
|
|
||||||
|
#endif /* _NLS_PRIVATE */
|
||||||
|
|
||||||
|
#define NL_SETD 0
|
||||||
|
#define NL_CAT_LOCALE 1
|
||||||
|
|
||||||
|
typedef struct __nl_cat_d {
|
||||||
|
void *__data;
|
||||||
|
int __size;
|
||||||
|
} *nl_catd;
|
||||||
|
|
||||||
|
|
||||||
|
#ifndef _NL_ITEM_DECLARED
|
||||||
|
typedef int nl_item;
|
||||||
|
#define _NL_ITEM_DECLARED
|
||||||
|
#endif
|
||||||
|
|
||||||
|
__BEGIN_DECLS
|
||||||
|
nl_catd catopen(const char *, int);
|
||||||
|
char *catgets(nl_catd, int, int, const char *) __format_arg(4);
|
||||||
|
int catclose(nl_catd);
|
||||||
|
__END_DECLS
|
||||||
|
|
||||||
|
#endif /* _NL_TYPES_H_ */
|
478
winsup/cygwin/libc/msgcat.c
Normal file
478
winsup/cygwin/libc/msgcat.c
Normal file
@ -0,0 +1,478 @@
|
|||||||
|
/***********************************************************
|
||||||
|
Copyright 1990, by Alfalfa Software Incorporated, Cambridge, Massachusetts.
|
||||||
|
Copyright 2010, Gabor Kovesdan <gabor@FreeBSD.org>
|
||||||
|
|
||||||
|
All Rights Reserved
|
||||||
|
|
||||||
|
Permission to use, copy, modify, and distribute this software and its
|
||||||
|
documentation for any purpose and without fee is hereby granted,
|
||||||
|
provided that the above copyright notice appear in all copies and that
|
||||||
|
both that copyright notice and this permission notice appear in
|
||||||
|
supporting documentation, and that Alfalfa's name not be used in
|
||||||
|
advertising or publicity pertaining to distribution of the software
|
||||||
|
without specific, written prior permission.
|
||||||
|
|
||||||
|
ALPHALPHA DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING
|
||||||
|
ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL
|
||||||
|
ALPHALPHA BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR
|
||||||
|
ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS,
|
||||||
|
WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION,
|
||||||
|
ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
|
||||||
|
SOFTWARE.
|
||||||
|
|
||||||
|
If you make any modifications, bugfixes or other changes to this software
|
||||||
|
we'd appreciate it if you could send a copy to us so we can keep things
|
||||||
|
up-to-date. Many thanks.
|
||||||
|
Kee Hinckley
|
||||||
|
Alfalfa Software, Inc.
|
||||||
|
267 Allston St., #3
|
||||||
|
Cambridge, MA 02139 USA
|
||||||
|
nazgul@alfalfa.com
|
||||||
|
|
||||||
|
******************************************************************/
|
||||||
|
|
||||||
|
#include <sys/cdefs.h>
|
||||||
|
__FBSDID("$FreeBSD$");
|
||||||
|
|
||||||
|
#define _NLS_PRIVATE
|
||||||
|
|
||||||
|
#ifndef __CYGWIN__
|
||||||
|
#include "namespace.h"
|
||||||
|
#endif
|
||||||
|
#include <sys/types.h>
|
||||||
|
#include <sys/stat.h>
|
||||||
|
#include <sys/mman.h>
|
||||||
|
#include <sys/queue.h>
|
||||||
|
|
||||||
|
#include <arpa/inet.h> /* for ntohl() */
|
||||||
|
|
||||||
|
#include <errno.h>
|
||||||
|
#include <fcntl.h>
|
||||||
|
#include <limits.h>
|
||||||
|
#include <nl_types.h>
|
||||||
|
#include <pthread.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#ifndef __CYGWIN__
|
||||||
|
#include "un-namespace.h"
|
||||||
|
|
||||||
|
#include "../locale/xlocale_private.h"
|
||||||
|
|
||||||
|
#define _DEFAULT_NLS_PATH "/usr/share/nls/%L/%N.cat:/usr/share/nls/%N/%L:/usr/local/share/nls/%L/%N.cat:/usr/local/share/nls/%N/%L"
|
||||||
|
|
||||||
|
#define RLOCK(fail) { int ret; \
|
||||||
|
if (__isthreaded && \
|
||||||
|
((ret = _pthread_rwlock_rdlock(&rwlock)) != 0)) { \
|
||||||
|
errno = ret; \
|
||||||
|
return (fail); \
|
||||||
|
}}
|
||||||
|
#define WLOCK(fail) { int ret; \
|
||||||
|
if (__isthreaded && \
|
||||||
|
((ret = _pthread_rwlock_wrlock(&rwlock)) != 0)) { \
|
||||||
|
errno = ret; \
|
||||||
|
return (fail); \
|
||||||
|
}}
|
||||||
|
#define UNLOCK { if (__isthreaded) \
|
||||||
|
_pthread_rwlock_unlock(&rwlock); }
|
||||||
|
|
||||||
|
static pthread_rwlock_t rwlock = PTHREAD_RWLOCK_INITIALIZER;
|
||||||
|
#else
|
||||||
|
|
||||||
|
#include "../locale/setlocale.h"
|
||||||
|
#define SIZE_T_MAX __SIZE_MAX__
|
||||||
|
#define _close close
|
||||||
|
#define _open open
|
||||||
|
#define _fstat fstat
|
||||||
|
#define RLOCK(a)
|
||||||
|
#define WLOCK(a)
|
||||||
|
#define UNLOCK
|
||||||
|
|
||||||
|
#define _DEFAULT_NLS_PATH "/usr/share/locale/%L/%N.cat:/usr/share/locale/%L/LC_MESSAGES/%N.cat:/usr/share/locale/%L/%N:/usr/share/locale/%L/LC_MESSAGES/%N:/usr/share/locale/%l/%N.cat:/usr/share/locale/%l/LC_MESSAGES/%N.cat:/usr/share/locale/%l/%N:/usr/share/locale/%l/LC_MESSAGES/%N"
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#define NLERR ((nl_catd) -1)
|
||||||
|
#define NLRETERR(errc) { errno = errc; return (NLERR); }
|
||||||
|
#define SAVEFAIL(n, l, e) { WLOCK(NLERR); \
|
||||||
|
np = malloc(sizeof(struct catentry)); \
|
||||||
|
if (np != NULL) { \
|
||||||
|
np->name = strdup(n); \
|
||||||
|
np->path = NULL; \
|
||||||
|
np->catd = NLERR; \
|
||||||
|
np->refcount = 0; \
|
||||||
|
np->lang = (l == NULL) ? NULL : \
|
||||||
|
strdup(l); \
|
||||||
|
np->caterrno = e; \
|
||||||
|
SLIST_INSERT_HEAD(&cache, np, list); \
|
||||||
|
} \
|
||||||
|
UNLOCK; \
|
||||||
|
errno = e; \
|
||||||
|
}
|
||||||
|
|
||||||
|
static nl_catd load_msgcat(const char *, const char *, const char *);
|
||||||
|
|
||||||
|
struct catentry {
|
||||||
|
SLIST_ENTRY(catentry) list;
|
||||||
|
char *name;
|
||||||
|
char *path;
|
||||||
|
int caterrno;
|
||||||
|
nl_catd catd;
|
||||||
|
char *lang;
|
||||||
|
int refcount;
|
||||||
|
};
|
||||||
|
|
||||||
|
static SLIST_HEAD(listhead, catentry) cache =
|
||||||
|
SLIST_HEAD_INITIALIZER(cache);
|
||||||
|
|
||||||
|
nl_catd
|
||||||
|
catopen(const char *name, int type)
|
||||||
|
{
|
||||||
|
struct stat sbuf;
|
||||||
|
struct catentry *np;
|
||||||
|
char *base, *cptr, *cptr1, *nlspath, *pathP, *pcode;
|
||||||
|
char *plang, *pter;
|
||||||
|
int saverr, spcleft;
|
||||||
|
const char *lang, *tmpptr;
|
||||||
|
char path[PATH_MAX];
|
||||||
|
|
||||||
|
/* sanity checking */
|
||||||
|
if (name == NULL || *name == '\0')
|
||||||
|
NLRETERR(EINVAL);
|
||||||
|
|
||||||
|
if (strchr(name, '/') != NULL)
|
||||||
|
/* have a pathname */
|
||||||
|
lang = NULL;
|
||||||
|
else {
|
||||||
|
if (type == NL_CAT_LOCALE)
|
||||||
|
#ifdef __CYGWIN__
|
||||||
|
lang = __get_current_locale()->categories[LC_MESSAGES];
|
||||||
|
#else
|
||||||
|
lang = querylocale(LC_MESSAGES_MASK, __get_locale());
|
||||||
|
#endif
|
||||||
|
else
|
||||||
|
lang = getenv("LANG");
|
||||||
|
|
||||||
|
if (lang == NULL || *lang == '\0' || strlen(lang) > ENCODING_LEN ||
|
||||||
|
(lang[0] == '.' &&
|
||||||
|
(lang[1] == '\0' || (lang[1] == '.' && lang[2] == '\0'))) ||
|
||||||
|
strchr(lang, '/') != NULL)
|
||||||
|
lang = "C";
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Try to get it from the cache first */
|
||||||
|
RLOCK(NLERR);
|
||||||
|
SLIST_FOREACH(np, &cache, list) {
|
||||||
|
if ((strcmp(np->name, name) == 0) &&
|
||||||
|
((lang != NULL && np->lang != NULL &&
|
||||||
|
strcmp(np->lang, lang) == 0) || (np->lang == lang))) {
|
||||||
|
if (np->caterrno != 0) {
|
||||||
|
/* Found cached failing entry */
|
||||||
|
UNLOCK;
|
||||||
|
NLRETERR(np->caterrno);
|
||||||
|
} else {
|
||||||
|
/* Found cached successful entry */
|
||||||
|
np->refcount++;
|
||||||
|
UNLOCK;
|
||||||
|
return (np->catd);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
UNLOCK;
|
||||||
|
|
||||||
|
/* is it absolute path ? if yes, load immediately */
|
||||||
|
if (strchr(name, '/') != NULL)
|
||||||
|
return (load_msgcat(name, name, lang));
|
||||||
|
|
||||||
|
/* sanity checking */
|
||||||
|
if ((plang = cptr1 = strdup(lang)) == NULL)
|
||||||
|
return (NLERR);
|
||||||
|
if ((cptr = strchr(cptr1, '@')) != NULL)
|
||||||
|
*cptr = '\0';
|
||||||
|
pter = pcode = (char *)"";
|
||||||
|
if ((cptr = strchr(cptr1, '_')) != NULL) {
|
||||||
|
*cptr++ = '\0';
|
||||||
|
pter = cptr1 = cptr;
|
||||||
|
}
|
||||||
|
if ((cptr = strchr(cptr1, '.')) != NULL) {
|
||||||
|
*cptr++ = '\0';
|
||||||
|
pcode = cptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((nlspath = getenv("NLSPATH")) == NULL || issetugid())
|
||||||
|
nlspath = (char *)_DEFAULT_NLS_PATH;
|
||||||
|
|
||||||
|
if ((base = cptr = strdup(nlspath)) == NULL) {
|
||||||
|
saverr = errno;
|
||||||
|
free(plang);
|
||||||
|
errno = saverr;
|
||||||
|
return (NLERR);
|
||||||
|
}
|
||||||
|
|
||||||
|
while ((nlspath = strsep(&cptr, ":")) != NULL) {
|
||||||
|
pathP = path;
|
||||||
|
if (*nlspath) {
|
||||||
|
for (; *nlspath; ++nlspath) {
|
||||||
|
if (*nlspath == '%') {
|
||||||
|
switch (*(nlspath + 1)) {
|
||||||
|
case 'l':
|
||||||
|
tmpptr = plang;
|
||||||
|
break;
|
||||||
|
case 't':
|
||||||
|
tmpptr = pter;
|
||||||
|
break;
|
||||||
|
case 'c':
|
||||||
|
tmpptr = pcode;
|
||||||
|
break;
|
||||||
|
case 'L':
|
||||||
|
tmpptr = lang;
|
||||||
|
break;
|
||||||
|
case 'N':
|
||||||
|
tmpptr = (char *)name;
|
||||||
|
break;
|
||||||
|
case '%':
|
||||||
|
++nlspath;
|
||||||
|
/* FALLTHROUGH */
|
||||||
|
default:
|
||||||
|
if (pathP - path >=
|
||||||
|
sizeof(path) - 1)
|
||||||
|
goto too_long;
|
||||||
|
*(pathP++) = *nlspath;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
++nlspath;
|
||||||
|
put_tmpptr:
|
||||||
|
spcleft = sizeof(path) -
|
||||||
|
(pathP - path) - 1;
|
||||||
|
if (strlcpy(pathP, tmpptr, spcleft) >=
|
||||||
|
spcleft) {
|
||||||
|
too_long:
|
||||||
|
free(plang);
|
||||||
|
free(base);
|
||||||
|
SAVEFAIL(name, lang, ENAMETOOLONG);
|
||||||
|
NLRETERR(ENAMETOOLONG);
|
||||||
|
}
|
||||||
|
pathP += strlen(tmpptr);
|
||||||
|
} else {
|
||||||
|
if (pathP - path >= sizeof(path) - 1)
|
||||||
|
goto too_long;
|
||||||
|
*(pathP++) = *nlspath;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
*pathP = '\0';
|
||||||
|
if (stat(path, &sbuf) == 0) {
|
||||||
|
free(plang);
|
||||||
|
free(base);
|
||||||
|
return (load_msgcat(path, name, lang));
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
tmpptr = (char *)name;
|
||||||
|
--nlspath;
|
||||||
|
goto put_tmpptr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
free(plang);
|
||||||
|
free(base);
|
||||||
|
SAVEFAIL(name, lang, ENOENT);
|
||||||
|
NLRETERR(ENOENT);
|
||||||
|
}
|
||||||
|
|
||||||
|
char *
|
||||||
|
catgets(nl_catd catd, int set_id, int msg_id, const char *s)
|
||||||
|
{
|
||||||
|
struct _nls_cat_hdr *cat_hdr;
|
||||||
|
struct _nls_msg_hdr *msg_hdr;
|
||||||
|
struct _nls_set_hdr *set_hdr;
|
||||||
|
int i, l, r, u;
|
||||||
|
|
||||||
|
if (catd == NULL || catd == NLERR) {
|
||||||
|
errno = EBADF;
|
||||||
|
/* LINTED interface problem */
|
||||||
|
return ((char *)s);
|
||||||
|
}
|
||||||
|
|
||||||
|
cat_hdr = (struct _nls_cat_hdr *)catd->__data;
|
||||||
|
set_hdr = (struct _nls_set_hdr *)(void *)((char *)catd->__data +
|
||||||
|
sizeof(struct _nls_cat_hdr));
|
||||||
|
|
||||||
|
/* binary search, see knuth algorithm b */
|
||||||
|
l = 0;
|
||||||
|
u = ntohl((u_int32_t)cat_hdr->__nsets) - 1;
|
||||||
|
while (l <= u) {
|
||||||
|
i = (l + u) / 2;
|
||||||
|
r = set_id - ntohl((u_int32_t)set_hdr[i].__setno);
|
||||||
|
|
||||||
|
if (r == 0) {
|
||||||
|
msg_hdr = (struct _nls_msg_hdr *)
|
||||||
|
(void *)((char *)catd->__data +
|
||||||
|
sizeof(struct _nls_cat_hdr) +
|
||||||
|
ntohl((u_int32_t)cat_hdr->__msg_hdr_offset));
|
||||||
|
|
||||||
|
l = ntohl((u_int32_t)set_hdr[i].__index);
|
||||||
|
u = l + ntohl((u_int32_t)set_hdr[i].__nmsgs) - 1;
|
||||||
|
while (l <= u) {
|
||||||
|
i = (l + u) / 2;
|
||||||
|
r = msg_id -
|
||||||
|
ntohl((u_int32_t)msg_hdr[i].__msgno);
|
||||||
|
if (r == 0) {
|
||||||
|
return ((char *) catd->__data +
|
||||||
|
sizeof(struct _nls_cat_hdr) +
|
||||||
|
ntohl((u_int32_t)
|
||||||
|
cat_hdr->__msg_txt_offset) +
|
||||||
|
ntohl((u_int32_t)
|
||||||
|
msg_hdr[i].__offset));
|
||||||
|
} else if (r < 0) {
|
||||||
|
u = i - 1;
|
||||||
|
} else {
|
||||||
|
l = i + 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* not found */
|
||||||
|
goto notfound;
|
||||||
|
|
||||||
|
} else if (r < 0) {
|
||||||
|
u = i - 1;
|
||||||
|
} else {
|
||||||
|
l = i + 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
notfound:
|
||||||
|
/* not found */
|
||||||
|
errno = ENOMSG;
|
||||||
|
/* LINTED interface problem */
|
||||||
|
return ((char *)s);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
catfree(struct catentry *np)
|
||||||
|
{
|
||||||
|
|
||||||
|
if (np->catd != NULL && np->catd != NLERR) {
|
||||||
|
munmap(np->catd->__data, (size_t)np->catd->__size);
|
||||||
|
free(np->catd);
|
||||||
|
}
|
||||||
|
SLIST_REMOVE(&cache, np, catentry, list);
|
||||||
|
free(np->name);
|
||||||
|
free(np->path);
|
||||||
|
free(np->lang);
|
||||||
|
free(np);
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
catclose(nl_catd catd)
|
||||||
|
{
|
||||||
|
struct catentry *np;
|
||||||
|
|
||||||
|
/* sanity checking */
|
||||||
|
if (catd == NULL || catd == NLERR) {
|
||||||
|
errno = EBADF;
|
||||||
|
return (-1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Remove from cache if not referenced any more */
|
||||||
|
WLOCK(-1);
|
||||||
|
SLIST_FOREACH(np, &cache, list) {
|
||||||
|
if (catd == np->catd) {
|
||||||
|
np->refcount--;
|
||||||
|
if (np->refcount == 0)
|
||||||
|
catfree(np);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
UNLOCK;
|
||||||
|
return (0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Internal support functions
|
||||||
|
*/
|
||||||
|
|
||||||
|
static nl_catd
|
||||||
|
load_msgcat(const char *path, const char *name, const char *lang)
|
||||||
|
{
|
||||||
|
struct stat st;
|
||||||
|
nl_catd catd;
|
||||||
|
struct catentry *np;
|
||||||
|
void *data;
|
||||||
|
int fd;
|
||||||
|
|
||||||
|
/* path/name will never be NULL here */
|
||||||
|
|
||||||
|
/*
|
||||||
|
* One more try in cache; if it was not found by name,
|
||||||
|
* it might still be found by absolute path.
|
||||||
|
*/
|
||||||
|
RLOCK(NLERR);
|
||||||
|
SLIST_FOREACH(np, &cache, list) {
|
||||||
|
if ((np->path != NULL) && (strcmp(np->path, path) == 0)) {
|
||||||
|
np->refcount++;
|
||||||
|
UNLOCK;
|
||||||
|
return (np->catd);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
UNLOCK;
|
||||||
|
|
||||||
|
if ((fd = _open(path, O_RDONLY | O_CLOEXEC)) == -1) {
|
||||||
|
SAVEFAIL(name, lang, errno);
|
||||||
|
NLRETERR(errno);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (_fstat(fd, &st) != 0) {
|
||||||
|
_close(fd);
|
||||||
|
SAVEFAIL(name, lang, EFTYPE);
|
||||||
|
NLRETERR(EFTYPE);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If the file size cannot be held in size_t we cannot mmap()
|
||||||
|
* it to the memory. Probably, this will not be a problem given
|
||||||
|
* that catalog files are usually small.
|
||||||
|
*/
|
||||||
|
if (st.st_size > SIZE_T_MAX) {
|
||||||
|
_close(fd);
|
||||||
|
SAVEFAIL(name, lang, EFBIG);
|
||||||
|
NLRETERR(EFBIG);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((data = mmap(0, (size_t)st.st_size, PROT_READ,
|
||||||
|
MAP_FILE|MAP_SHARED, fd, (off_t)0)) == MAP_FAILED) {
|
||||||
|
int saved_errno = errno;
|
||||||
|
_close(fd);
|
||||||
|
SAVEFAIL(name, lang, saved_errno);
|
||||||
|
NLRETERR(saved_errno);
|
||||||
|
}
|
||||||
|
_close(fd);
|
||||||
|
|
||||||
|
if (ntohl((u_int32_t)((struct _nls_cat_hdr *)data)->__magic) !=
|
||||||
|
_NLS_MAGIC) {
|
||||||
|
munmap(data, (size_t)st.st_size);
|
||||||
|
SAVEFAIL(name, lang, EFTYPE);
|
||||||
|
NLRETERR(EFTYPE);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((catd = malloc(sizeof (*catd))) == NULL) {
|
||||||
|
munmap(data, (size_t)st.st_size);
|
||||||
|
SAVEFAIL(name, lang, ENOMEM);
|
||||||
|
NLRETERR(ENOMEM);
|
||||||
|
}
|
||||||
|
|
||||||
|
catd->__data = data;
|
||||||
|
catd->__size = (int)st.st_size;
|
||||||
|
|
||||||
|
/* Caching opened catalog */
|
||||||
|
WLOCK(NLERR);
|
||||||
|
if ((np = malloc(sizeof(struct catentry))) != NULL) {
|
||||||
|
np->name = strdup(name);
|
||||||
|
np->path = strdup(path);
|
||||||
|
np->catd = catd;
|
||||||
|
np->lang = (lang == NULL) ? NULL : strdup(lang);
|
||||||
|
np->refcount = 1;
|
||||||
|
np->caterrno = 0;
|
||||||
|
SLIST_INSERT_HEAD(&cache, np, list);
|
||||||
|
}
|
||||||
|
UNLOCK;
|
||||||
|
return (catd);
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user