diff options
Diffstat (limited to 'libc-top-half/musl/src/locale/catopen.c')
-rw-r--r-- | libc-top-half/musl/src/locale/catopen.c | 87 |
1 files changed, 87 insertions, 0 deletions
diff --git a/libc-top-half/musl/src/locale/catopen.c b/libc-top-half/musl/src/locale/catopen.c new file mode 100644 index 0000000..c6116b1 --- /dev/null +++ b/libc-top-half/musl/src/locale/catopen.c @@ -0,0 +1,87 @@ +#define _BSD_SOURCE +#include <nl_types.h> +#include <string.h> +#include <stdint.h> +#include <endian.h> +#include <errno.h> +#include <langinfo.h> +#include <locale.h> +#ifdef __wasilibc_unmodified_upstream // wasi-libc doesn't support catgets yet +#include <sys/mman.h> +#endif +#include "libc.h" + +#define V(p) be32toh(*(uint32_t *)(p)) + +#ifdef __wasilibc_unmodified_upstream // wasi-libc doesn't support catgets yet +static nl_catd do_catopen(const char *name) +{ + size_t size; + const unsigned char *map = __map_file(name, &size); + /* Size recorded in the file must match file size; otherwise + * the information needed to unmap the file will be lost. */ + if (!map || V(map) != 0xff88ff89 || 20+V(map+8) != size) { + if(map) munmap((void *)map, size); + errno = ENOENT; + return (nl_catd)-1; + } + return (nl_catd)map; +} +#endif + +nl_catd catopen(const char *name, int oflag) +{ +#ifdef __wasilibc_unmodified_upstream // wasi-libc doesn't support catgets yet + nl_catd catd; + + if (strchr(name, '/')) return do_catopen(name); + + char buf[PATH_MAX]; + size_t i; + const char *path, *lang, *p, *z; + if (libc.secure || !(path = getenv("NLSPATH"))) { + errno = ENOENT; + return (nl_catd)-1; + } + lang = oflag ? nl_langinfo(_NL_LOCALE_NAME(LC_MESSAGES)) : getenv("LANG"); + if (!lang) lang = ""; + for (p=path; *p; p=z) { + i = 0; + z = __strchrnul(p, ':'); + for (; p<z; p++) { + const char *v; + size_t l; + if (*p!='%') v=p, l=1; + else switch (*++p) { + case 'N': v=name; l=strlen(v); break; + case 'L': v=lang; l=strlen(v); break; + case 'l': v=lang; l=strcspn(v,"_.@"); break; + case 't': + v=__strchrnul(lang,'_'); + if (*v) v++; + l=strcspn(v,".@"); + break; + case 'c': v="UTF-8"; l=5; break; + case '%': v="%"; l=1; break; + default: v=0; + } + if (!v || l >= sizeof buf - i) { + break; + } + memcpy(buf+i, v, l); + i += l; + } + if (!*z && (p<z || !i)) break; + if (p<z) continue; + if (*z) z++; + buf[i] = 0; + /* Leading : or :: in NLSPATH is same as %N */ + catd = do_catopen(i ? buf : name); + if (catd != (nl_catd)-1) return catd; + } + errno = ENOENT; +#else + errno = EOPNOTSUPP; +#endif + return (nl_catd)-1; +} |