29#include <gnu/libc-version.h>
42#define TEXTBUFLEN 1024
46extern size_t strlower_libc(
char *dst,
size_t dstsize,
const char *src,
48extern size_t strtitle_libc(
char *dst,
size_t dstsize,
const char *src,
50extern size_t strupper_libc(
char *dst,
size_t dstsize,
const char *src,
54 const char *arg2, ssize_t len2,
57 const char *src, ssize_t srclen,
65static int strncoll_libc_win32_utf8(
const char *arg1, ssize_t len1,
66 const char *arg2, ssize_t len2,
71 const char *src, ssize_t srclen,
74 const char *src, ssize_t srclen,
77 const char *src, ssize_t srclen,
80 const char *src, ssize_t srclen,
83 const char *src, ssize_t srclen,
86 const char *src, ssize_t srclen,
92 .strnxfrm_prefix = NULL,
104 .strxfrm_is_safe =
true,
106 .strxfrm_is_safe =
false,
112 .
strncoll = strncoll_libc_win32_utf8,
114 .strnxfrm_prefix = NULL,
116 .strxfrm_is_safe =
true,
118 .strxfrm_is_safe =
false,
158 srclen = strlen(src);
160 if (srclen + 1 <= destsize)
165 if (srclen + 1 > destsize)
168 memcpy(
dest, src, srclen);
178 for (p =
dest; *p; p++)
202 srclen = strlen(src);
205 if ((srclen + 1) > (INT_MAX /
sizeof(
wchar_t)))
207 (
errcode(ERRCODE_OUT_OF_MEMORY),
208 errmsg(
"out of memory")));
211 workspace = (
wchar_t *)
palloc((srclen + 1) *
sizeof(wchar_t));
215 for (curr_char = 0; workspace[curr_char] != 0; curr_char++)
216 workspace[curr_char] =
towlower_l(workspace[curr_char], loc);
222 result =
palloc(max_size + 1);
226 if (result_size + 1 > destsize)
229 memcpy(
dest, result, result_size);
230 dest[result_size] =
'\0';
243 srclen = strlen(src);
245 if (srclen + 1 <= destsize)
248 int wasalnum =
false;
251 memcpy(
dest, src, srclen);
261 for (p =
dest; *p; p++)
277 wasalnum =
isalnum_l((
unsigned char) *p, loc);
289 int wasalnum =
false;
297 srclen = strlen(src);
300 if ((srclen + 1) > (INT_MAX /
sizeof(
wchar_t)))
302 (
errcode(ERRCODE_OUT_OF_MEMORY),
303 errmsg(
"out of memory")));
306 workspace = (
wchar_t *)
palloc((srclen + 1) *
sizeof(wchar_t));
310 for (curr_char = 0; workspace[curr_char] != 0; curr_char++)
313 workspace[curr_char] =
towlower_l(workspace[curr_char], loc);
315 workspace[curr_char] =
towupper_l(workspace[curr_char], loc);
316 wasalnum =
iswalnum_l(workspace[curr_char], loc);
323 result =
palloc(max_size + 1);
327 if (result_size + 1 > destsize)
330 memcpy(
dest, result, result_size);
331 dest[result_size] =
'\0';
344 srclen = strlen(src);
346 if (srclen + 1 <= destsize)
351 memcpy(
dest, src, srclen);
361 for (p =
dest; *p; p++)
385 srclen = strlen(src);
388 if ((srclen + 1) > (INT_MAX /
sizeof(
wchar_t)))
390 (
errcode(ERRCODE_OUT_OF_MEMORY),
391 errmsg(
"out of memory")));
394 workspace = (
wchar_t *)
palloc((srclen + 1) *
sizeof(wchar_t));
398 for (curr_char = 0; workspace[curr_char] != 0; curr_char++)
399 workspace[curr_char] =
towupper_l(workspace[curr_char], loc);
405 result =
palloc(max_size + 1);
409 if (result_size + 1 > destsize)
412 memcpy(
dest, result, result_size);
413 dest[result_size] =
'\0';
429 if (
collid == DEFAULT_COLLATION_OID)
438 Anum_pg_database_datcollate);
441 Anum_pg_database_datctype);
456 Anum_pg_collation_collcollate);
459 Anum_pg_collation_collctype);
469 result->
provider = COLLPROVIDER_LIBC;
472 (strcmp(collate,
"POSIX") == 0);
473 result->
ctype_is_c = (strcmp(ctype,
"C") == 0) ||
474 (strcmp(ctype,
"POSIX") == 0);
480 result->
collate = &collate_methods_libc_win32_utf8;
502 if (strcmp(collate, ctype) == 0)
504 if (strcmp(ctype,
"C") != 0 && strcmp(ctype,
"POSIX") != 0)
509 loc = newlocale(LC_COLLATE_MASK | LC_CTYPE_MASK, collate,
512 loc = _create_locale(LC_ALL, collate);
524 if (strcmp(collate,
"C") != 0 && strcmp(collate,
"POSIX") != 0)
527 loc1 = newlocale(LC_COLLATE_MASK, collate, NULL);
532 if (strcmp(ctype,
"C") != 0 && strcmp(ctype,
"POSIX") != 0)
535 loc = newlocale(LC_CTYPE_MASK, ctype, loc1);
553 (
errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
554 errmsg(
"collations with different collate and ctype values are not supported on this platform")));
569strncoll_libc(
const char *arg1, ssize_t len1,
const char *arg2, ssize_t len2,
574 size_t bufsize1 = (len1 == -1) ? 0 : len1 + 1;
575 size_t bufsize2 = (len2 == -1) ? 0 : len2 + 1;
594 memcpy(buf1, arg1, len1);
605 char *buf2 =
buf + bufsize1;
607 memcpy(buf2, arg2, len2);
645 memcpy(
buf, src, srclen);
654 Assert(result >= destsize ||
dest[result] ==
'\0');
662 char *collversion = NULL;
668#if defined(__GLIBC__)
670 collversion =
pstrdup(gnu_get_libc_version());
671#elif defined(LC_VERSION_MASK)
675 loc = newlocale(LC_COLLATE_MASK, collcollate, NULL);
679 pstrdup(querylocale(LC_COLLATE_MASK | LC_VERSION_MASK, loc));
684 (
errmsg(
"could not load locale \"%s\"", collcollate)));
691 NLSVERSIONINFOEX version = {
sizeof(NLSVERSIONINFOEX)};
692 WCHAR wide_collcollate[LOCALE_NAME_MAX_LENGTH];
694 MultiByteToWideChar(CP_ACP, 0, collcollate, -1, wide_collcollate,
695 LOCALE_NAME_MAX_LENGTH);
696 if (!GetNLSVersionEx(COMPARE_STRING, wide_collcollate, &version))
705 if (GetLastError() == ERROR_INVALID_PARAMETER)
709 (
errmsg(
"could not get collation version for locale \"%s\": error code %lu",
713 collversion =
psprintf(
"%lu.%lu,%lu.%lu",
714 (version.dwNLSVersion >> 8) & 0xFFFF,
715 version.dwNLSVersion & 0xFF,
716 (version.dwDefinedVersion >> 8) & 0xFFFF,
717 version.dwDefinedVersion & 0xFF);
734strncoll_libc_win32_utf8(
const char *arg1, ssize_t len1,
const char *arg2,
754 a1len = len1 * 2 + 2;
755 a2len = len2 * 2 + 2;
768 r = MultiByteToWideChar(CP_UTF8, 0, arg1, len1,
769 (LPWSTR) a1p, a1len / 2);
772 (
errmsg(
"could not convert string to UTF-16: error code %lu",
775 ((LPWSTR) a1p)[r] = 0;
781 r = MultiByteToWideChar(CP_UTF8, 0, arg2, len2,
782 (LPWSTR) a2p, a2len / 2);
785 (
errmsg(
"could not convert string to UTF-16: error code %lu",
788 ((LPWSTR) a2p)[r] = 0;
792 if (result == 2147483647)
794 (
errmsg(
"could not compare Unicode strings: %m")));
825 (
errcode(ERRCODE_INVALID_PARAMETER_VALUE),
826 errmsg(
"could not create locale \"%s\": %m",
828 (save_errno == ENOENT ?
829 errdetail(
"The operating system could not find any locale data for the locale name \"%s\".",
837#ifndef HAVE_MBSTOWCS_L
842 return _mbstowcs_l(
dest, src, n, loc);
845 locale_t save_locale = uselocale(loc);
847 result = mbstowcs(
dest, src, n);
848 uselocale(save_locale);
853#ifndef HAVE_WCSTOMBS_L
858 return _wcstombs_l(
dest, src, n, loc);
861 locale_t save_locale = uselocale(loc);
863 result = wcstombs(
dest, src, n);
864 uselocale(save_locale);
899 result = WideCharToMultiByte(CP_UTF8, 0, from, -1, to, tolen,
916 result = wcstombs(to, from, tolen);
937char2wchar(
wchar_t *to,
size_t tolen,
const char *from,
size_t fromlen,
954 result = MultiByteToWideChar(CP_UTF8, 0, from, fromlen, to, tolen - 1);
976 result = mbstowcs(to,
str, tolen);
1000 (
errcode(ERRCODE_CHARACTER_NOT_IN_REPERTOIRE),
1001 errmsg(
"invalid multibyte character for locale"),
1002 errhint(
"The server's LC_CTYPE locale is probably incompatible with the database encoding.")));
#define TextDatumGetCString(d)
int errdetail(const char *fmt,...)
int errhint(const char *fmt,...)
int errcode(int sqlerrcode)
int errmsg(const char *fmt,...)
#define ereport(elevel,...)
Assert(PointerIsAligned(start, uint64))
#define HeapTupleIsValid(tuple)
int GetDatabaseEncoding(void)
bool pg_verifymbstr(const char *mbstr, int len, bool noError)
int pg_database_encoding_max_length(void)
void * MemoryContextAllocZero(MemoryContext context, Size size)
char * pstrdup(const char *in)
void pfree(void *pointer)
char * pnstrdup(const char *in, Size len)
static size_t strtitle_libc_mb(char *dest, size_t destsize, const char *src, ssize_t srclen, pg_locale_t locale)
size_t wchar2char(char *to, const wchar_t *from, size_t tolen, pg_locale_t locale)
static size_t strupper_libc_sb(char *dest, size_t destsize, const char *src, ssize_t srclen, pg_locale_t locale)
pg_locale_t create_pg_locale_libc(Oid collid, MemoryContext context)
size_t strtitle_libc(char *dst, size_t dstsize, const char *src, ssize_t srclen, pg_locale_t locale)
size_t strupper_libc(char *dst, size_t dstsize, const char *src, ssize_t srclen, pg_locale_t locale)
char * get_collation_actual_version_libc(const char *collcollate)
static locale_t make_libc_collator(const char *collate, const char *ctype)
static size_t wcstombs_l(char *dest, const wchar_t *src, size_t n, locale_t loc)
static const struct collate_methods collate_methods_libc
size_t strlower_libc(char *dst, size_t dstsize, const char *src, ssize_t srclen, pg_locale_t locale)
static size_t strupper_libc_mb(char *dest, size_t destsize, const char *src, ssize_t srclen, pg_locale_t locale)
static int strncoll_libc(const char *arg1, ssize_t len1, const char *arg2, ssize_t len2, pg_locale_t locale)
static size_t strlower_libc_sb(char *dest, size_t destsize, const char *src, ssize_t srclen, pg_locale_t locale)
static void report_newlocale_failure(const char *localename)
static size_t strnxfrm_libc(char *dest, size_t destsize, const char *src, ssize_t srclen, pg_locale_t locale)
static size_t strtitle_libc_sb(char *dest, size_t destsize, const char *src, ssize_t srclen, pg_locale_t locale)
size_t char2wchar(wchar_t *to, size_t tolen, const char *from, size_t fromlen, pg_locale_t locale)
static size_t mbstowcs_l(wchar_t *dest, const char *src, size_t n, locale_t loc)
static size_t strlower_libc_mb(char *dest, size_t destsize, const char *src, ssize_t srclen, pg_locale_t locale)
int pg_strcasecmp(const char *s1, const char *s2)
unsigned char pg_toupper(unsigned char ch)
unsigned char pg_tolower(unsigned char ch)
int pg_strncasecmp(const char *s1, const char *s2, size_t n)
static Datum ObjectIdGetDatum(Oid X)
char * psprintf(const char *fmt,...)
int(* strncoll)(const char *arg1, ssize_t len1, const char *arg2, ssize_t len2, pg_locale_t locale)
const struct collate_methods * collate
union pg_locale_struct::@158 info
void ReleaseSysCache(HeapTuple tuple)
HeapTuple SearchSysCache1(int cacheId, Datum key1)
Datum SysCacheGetAttrNotNull(int cacheId, HeapTuple tup, AttrNumber attributeNumber)