Print this page
2964 need POSIX 2008 locale object support
@@ -1,10 +1,15 @@
/*
* Copyright 2010 Nexenta Systems, Inc. All rights reserved.
* Copyright (c) 2001 Alexey Zelkin <phantom@FreeBSD.org>
* All rights reserved.
*
+ * Copyright (c) 2011 The FreeBSD Foundation
+ * All rights reserved.
+ * Portions of this software were developed by David Chisnall
+ * under sponsorship from the FreeBSD Foundation.
+ *
* 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.
@@ -25,49 +30,87 @@
* SUCH DAMAGE.
*/
#include "lint.h"
#include <stddef.h>
+#include <stdlib.h>
#include "ldpart.h"
#include "lmessages.h"
#define LCMESSAGES_SIZE_FULL (sizeof (struct lc_messages_T) / sizeof (char *))
#define LCMESSAGES_SIZE_MIN \
(offsetof(struct lc_messages_T, yesstr) / sizeof (char *))
+struct xlocale_messages __xlocale_global_messages;
+
static char empty[] = "";
static const struct lc_messages_T _C_messages_locale = {
"^[yY]", /* yesexpr */
"^[nN]", /* noexpr */
"yes", /* yesstr */
"no" /* nostr */
};
-static struct lc_messages_T _messages_locale;
-static int _messages_using_locale;
-static char *_messages_locale_buf;
+static void
+destruct_messages(void *v)
+{
+ struct xlocale_messages *l = v;
-int
-__messages_load_locale(const char *name)
+ if (l->buffer != NULL)
+ free(l->buffer);
+
+ free(l);
+}
+
+static int
+messages_load_locale(struct xlocale_messages *loc, int *using_locale,
+ const char *name)
{
int ret;
+ struct lc_messages_T *l = &loc->locale;
- ret = __part_load_locale(name, &_messages_using_locale,
- &_messages_locale_buf, "LC_MESSAGES",
+ ret = __part_load_locale(name, using_locale,
+ &loc->buffer, "LC_MESSAGES",
LCMESSAGES_SIZE_FULL, LCMESSAGES_SIZE_MIN,
- (const char **)&_messages_locale);
+ (const char **)l);
if (ret == _LDP_LOADED) {
- if (_messages_locale.yesstr == NULL)
- _messages_locale.yesstr = empty;
- if (_messages_locale.nostr == NULL)
- _messages_locale.nostr = empty;
+ if (l->yesstr == NULL)
+ l->yesstr = empty;
+ if (l->nostr == NULL)
+ l->nostr = empty;
}
+
return (ret);
}
+int
+__messages_load_locale(const char *name)
+{
+ return (messages_load_locale(&__xlocale_global_messages,
+ &__xlocale_global_locale.using_messages_locale, name));
+}
+
+void *
+__messages_load(const char *name, locale_t l)
+{
+ struct xlocale_messages *new;
+
+ new = calloc(sizeof(struct xlocale_messages), 1);
+ /* XXX */
+ new->header.header.destructor = destruct_messages;
+ if (messages_load_locale(new, &l->using_messages_locale, name) ==
+ _LDP_ERROR) {
+ xlocale_release(new);
+ return (NULL);
+ }
+
+ return (new);
+}
+
struct lc_messages_T *
-__get_current_messages_locale(void)
+__get_current_messages_locale(locale_t loc)
{
- return (_messages_using_locale ? &_messages_locale :
- (struct lc_messages_T *)&_C_messages_locale);
+ return (loc->using_messages_locale
+ ? &((struct xlocale_messages *)loc->components[XLC_MESSAGES])->locale
+ : (struct lc_messages_T *)&_C_messages_locale);
}