xref: /dflybsd-src/usr.bin/sort/bwstring.c (revision 94400e62a03bc02c69070297b92ea9d2e22132c3)
150fc853eSJohn Marino /*-
250fc853eSJohn Marino  * Copyright (C) 2009 Gabor Kovesdan <gabor@FreeBSD.org>
350fc853eSJohn Marino  * Copyright (C) 2012 Oleg Moskalenko <mom040267@gmail.com>
450fc853eSJohn Marino  * All rights reserved.
550fc853eSJohn Marino  *
650fc853eSJohn Marino  * Redistribution and use in source and binary forms, with or without
750fc853eSJohn Marino  * modification, are permitted provided that the following conditions
850fc853eSJohn Marino  * are met:
950fc853eSJohn Marino  * 1. Redistributions of source code must retain the above copyright
1050fc853eSJohn Marino  *    notice, this list of conditions and the following disclaimer.
1150fc853eSJohn Marino  * 2. Redistributions in binary form must reproduce the above copyright
1250fc853eSJohn Marino  *    notice, this list of conditions and the following disclaimer in the
1350fc853eSJohn Marino  *    documentation and/or other materials provided with the distribution.
1450fc853eSJohn Marino  *
1550fc853eSJohn Marino  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
1650fc853eSJohn Marino  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
1750fc853eSJohn Marino  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
1850fc853eSJohn Marino  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
1950fc853eSJohn Marino  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
2050fc853eSJohn Marino  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
2150fc853eSJohn Marino  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
2250fc853eSJohn Marino  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
2350fc853eSJohn Marino  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
2450fc853eSJohn Marino  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
2550fc853eSJohn Marino  * SUCH DAMAGE.
2650fc853eSJohn Marino  *
2750fc853eSJohn Marino  * $FreeBSD: head/usr.bin/sort/bwstring.c 281181 2015-04-07 01:17:29Z pfg $
2850fc853eSJohn Marino  */
2950fc853eSJohn Marino 
3050fc853eSJohn Marino 
3150fc853eSJohn Marino #include <ctype.h>
3250fc853eSJohn Marino #include <errno.h>
3350fc853eSJohn Marino #include <err.h>
3450fc853eSJohn Marino #include <langinfo.h>
3550fc853eSJohn Marino #include <math.h>
3650fc853eSJohn Marino #include <stdlib.h>
3750fc853eSJohn Marino #include <string.h>
3850fc853eSJohn Marino #include <wchar.h>
3950fc853eSJohn Marino #include <wctype.h>
4050fc853eSJohn Marino 
4150fc853eSJohn Marino #include "bwstring.h"
4250fc853eSJohn Marino #include "sort.h"
4350fc853eSJohn Marino 
4450fc853eSJohn Marino bool byte_sort;
4550fc853eSJohn Marino 
4650fc853eSJohn Marino static wchar_t **wmonths;
4750fc853eSJohn Marino static unsigned char **cmonths;
4850fc853eSJohn Marino 
4950fc853eSJohn Marino /* initialise months */
5050fc853eSJohn Marino 
5150fc853eSJohn Marino void
initialise_months(void)5250fc853eSJohn Marino initialise_months(void)
5350fc853eSJohn Marino {
5450fc853eSJohn Marino 	const nl_item item[12] = { ABMON_1, ABMON_2, ABMON_3, ABMON_4,
5550fc853eSJohn Marino 	    ABMON_5, ABMON_6, ABMON_7, ABMON_8, ABMON_9, ABMON_10,
5650fc853eSJohn Marino 	    ABMON_11, ABMON_12 };
5750fc853eSJohn Marino 	unsigned char *tmp;
5850fc853eSJohn Marino 	size_t len;
5950fc853eSJohn Marino 
6050fc853eSJohn Marino 	if (MB_CUR_MAX == 1) {
6150fc853eSJohn Marino 		if (cmonths == NULL) {
6250fc853eSJohn Marino 			unsigned char *m;
6350fc853eSJohn Marino 
6450fc853eSJohn Marino 			cmonths = sort_malloc(sizeof(unsigned char*) * 12);
6550fc853eSJohn Marino 			for (int i = 0; i < 12; i++) {
6650fc853eSJohn Marino 				cmonths[i] = NULL;
6750fc853eSJohn Marino 				tmp = (unsigned char *) nl_langinfo(item[i]);
6850fc853eSJohn Marino 				if (debug_sort)
6950fc853eSJohn Marino 					printf("month[%d]=%s\n", i, tmp);
7050fc853eSJohn Marino 				if (*tmp == '\0')
7150fc853eSJohn Marino 					continue;
7250fc853eSJohn Marino 				m = sort_strdup(tmp);
7350fc853eSJohn Marino 				len = strlen(tmp);
7450fc853eSJohn Marino 				for (unsigned int j = 0; j < len; j++)
7550fc853eSJohn Marino 					m[j] = toupper(m[j]);
7650fc853eSJohn Marino 				cmonths[i] = m;
7750fc853eSJohn Marino 			}
7850fc853eSJohn Marino 		}
7950fc853eSJohn Marino 
8050fc853eSJohn Marino 	} else {
8150fc853eSJohn Marino 		if (wmonths == NULL) {
8250fc853eSJohn Marino 			wchar_t *m;
8350fc853eSJohn Marino 
8450fc853eSJohn Marino 			wmonths = sort_malloc(sizeof(wchar_t *) * 12);
8550fc853eSJohn Marino 			for (int i = 0; i < 12; i++) {
8650fc853eSJohn Marino 				wmonths[i] = NULL;
8750fc853eSJohn Marino 				tmp = (unsigned char *) nl_langinfo(item[i]);
8850fc853eSJohn Marino 				if (debug_sort)
8950fc853eSJohn Marino 					printf("month[%d]=%s\n", i, tmp);
9050fc853eSJohn Marino 				if (*tmp == '\0')
9150fc853eSJohn Marino 					continue;
9250fc853eSJohn Marino 				len = strlen(tmp);
9350fc853eSJohn Marino 				m = sort_malloc(SIZEOF_WCHAR_STRING(len + 1));
9450fc853eSJohn Marino 				if (mbstowcs(m, (char*)tmp, len) ==
9550fc853eSJohn Marino 				    ((size_t) - 1)) {
9650fc853eSJohn Marino 					sort_free(m);
9750fc853eSJohn Marino 					continue;
9850fc853eSJohn Marino 				}
9950fc853eSJohn Marino 				m[len] = L'\0';
10050fc853eSJohn Marino 				for (unsigned int j = 0; j < len; j++)
10150fc853eSJohn Marino 					m[j] = towupper(m[j]);
10250fc853eSJohn Marino 				wmonths[i] = m;
10350fc853eSJohn Marino 			}
10450fc853eSJohn Marino 		}
10550fc853eSJohn Marino 	}
10650fc853eSJohn Marino }
10750fc853eSJohn Marino 
10850fc853eSJohn Marino /*
10950fc853eSJohn Marino  * Compare two wide-character strings
11050fc853eSJohn Marino  */
11150fc853eSJohn Marino static int
wide_str_coll(const wchar_t * s1,const wchar_t * s2)11250fc853eSJohn Marino wide_str_coll(const wchar_t *s1, const wchar_t *s2)
11350fc853eSJohn Marino {
11450fc853eSJohn Marino 	int ret = 0;
11550fc853eSJohn Marino 
11650fc853eSJohn Marino 	errno = 0;
11750fc853eSJohn Marino 	ret = wcscoll(s1, s2);
11850fc853eSJohn Marino 	if (errno == EILSEQ) {
11950fc853eSJohn Marino 		errno = 0;
12050fc853eSJohn Marino 		ret = wcscmp(s1, s2);
12150fc853eSJohn Marino 		if (errno != 0) {
12250fc853eSJohn Marino 			for (size_t i = 0; ; ++i) {
12350fc853eSJohn Marino 				wchar_t c1 = s1[i];
12450fc853eSJohn Marino 				wchar_t c2 = s2[i];
12550fc853eSJohn Marino 				if (c1 == L'\0')
12650fc853eSJohn Marino 					return ((c2 == L'\0') ? 0 : -1);
12750fc853eSJohn Marino 				if (c2 == L'\0')
12850fc853eSJohn Marino 					return (+1);
12950fc853eSJohn Marino 				if (c1 == c2)
13050fc853eSJohn Marino 					continue;
13150fc853eSJohn Marino 				return ((int)(c1 - c2));
13250fc853eSJohn Marino 			}
13350fc853eSJohn Marino 		}
13450fc853eSJohn Marino 	}
13550fc853eSJohn Marino 	return (ret);
13650fc853eSJohn Marino }
13750fc853eSJohn Marino 
13850fc853eSJohn Marino /* counterparts of wcs functions */
13950fc853eSJohn Marino 
14050fc853eSJohn Marino void
bwsprintf(FILE * f,struct bwstring * bws,const char * prefix,const char * suffix)14150fc853eSJohn Marino bwsprintf(FILE *f, struct bwstring *bws, const char *prefix, const char *suffix)
14250fc853eSJohn Marino {
14350fc853eSJohn Marino 
14450fc853eSJohn Marino 	if (MB_CUR_MAX == 1)
14550fc853eSJohn Marino 		fprintf(f, "%s%s%s", prefix, bws->data.cstr, suffix);
14650fc853eSJohn Marino 	else
14750fc853eSJohn Marino 		fprintf(f, "%s%S%s", prefix, bws->data.wstr, suffix);
14850fc853eSJohn Marino }
14950fc853eSJohn Marino 
bwsrawdata(const struct bwstring * bws)15050fc853eSJohn Marino const void* bwsrawdata(const struct bwstring *bws)
15150fc853eSJohn Marino {
15250fc853eSJohn Marino 
15350fc853eSJohn Marino 	return (&(bws->data));
15450fc853eSJohn Marino }
15550fc853eSJohn Marino 
bwsrawlen(const struct bwstring * bws)15650fc853eSJohn Marino size_t bwsrawlen(const struct bwstring *bws)
15750fc853eSJohn Marino {
15850fc853eSJohn Marino 
15950fc853eSJohn Marino 	return ((MB_CUR_MAX == 1) ? bws->len : SIZEOF_WCHAR_STRING(bws->len));
16050fc853eSJohn Marino }
16150fc853eSJohn Marino 
16250fc853eSJohn Marino size_t
bws_memsize(const struct bwstring * bws)16350fc853eSJohn Marino bws_memsize(const struct bwstring *bws)
16450fc853eSJohn Marino {
16550fc853eSJohn Marino 
16650fc853eSJohn Marino 	return ((MB_CUR_MAX == 1) ? (bws->len + 2 + sizeof(struct bwstring)) :
16750fc853eSJohn Marino 	    (SIZEOF_WCHAR_STRING(bws->len + 1) + sizeof(struct bwstring)));
16850fc853eSJohn Marino }
16950fc853eSJohn Marino 
17050fc853eSJohn Marino void
bws_setlen(struct bwstring * bws,size_t newlen)17150fc853eSJohn Marino bws_setlen(struct bwstring *bws, size_t newlen)
17250fc853eSJohn Marino {
17350fc853eSJohn Marino 
17450fc853eSJohn Marino 	if (bws && newlen != bws->len && newlen <= bws->len) {
17550fc853eSJohn Marino 		bws->len = newlen;
17650fc853eSJohn Marino 		if (MB_CUR_MAX == 1)
17750fc853eSJohn Marino 			bws->data.cstr[newlen] = '\0';
17850fc853eSJohn Marino 		else
17950fc853eSJohn Marino 			bws->data.wstr[newlen] = L'\0';
18050fc853eSJohn Marino 	}
18150fc853eSJohn Marino }
18250fc853eSJohn Marino 
18350fc853eSJohn Marino /*
18450fc853eSJohn Marino  * Allocate a new binary string of specified size
18550fc853eSJohn Marino  */
18650fc853eSJohn Marino struct bwstring *
bwsalloc(size_t sz)18750fc853eSJohn Marino bwsalloc(size_t sz)
18850fc853eSJohn Marino {
18950fc853eSJohn Marino 	struct bwstring *ret;
19050fc853eSJohn Marino 
19150fc853eSJohn Marino 	if (MB_CUR_MAX == 1)
19250fc853eSJohn Marino 		ret = sort_malloc(sizeof(struct bwstring) + 1 + sz);
19350fc853eSJohn Marino 	else
19450fc853eSJohn Marino 		ret = sort_malloc(sizeof(struct bwstring) +
19550fc853eSJohn Marino 		    SIZEOF_WCHAR_STRING(sz + 1));
19650fc853eSJohn Marino 	ret->len = sz;
19750fc853eSJohn Marino 
19850fc853eSJohn Marino 	if (MB_CUR_MAX == 1)
19950fc853eSJohn Marino 		ret->data.cstr[ret->len] = '\0';
20050fc853eSJohn Marino 	else
20150fc853eSJohn Marino 		ret->data.wstr[ret->len] = L'\0';
20250fc853eSJohn Marino 
20350fc853eSJohn Marino 	return (ret);
20450fc853eSJohn Marino }
20550fc853eSJohn Marino 
20650fc853eSJohn Marino /*
20750fc853eSJohn Marino  * Create a copy of binary string.
20850fc853eSJohn Marino  * New string size equals the length of the old string.
20950fc853eSJohn Marino  */
21050fc853eSJohn Marino struct bwstring *
bwsdup(const struct bwstring * s)21150fc853eSJohn Marino bwsdup(const struct bwstring *s)
21250fc853eSJohn Marino {
21350fc853eSJohn Marino 
21450fc853eSJohn Marino 	if (s == NULL)
21550fc853eSJohn Marino 		return (NULL);
21650fc853eSJohn Marino 	else {
21750fc853eSJohn Marino 		struct bwstring *ret = bwsalloc(s->len);
21850fc853eSJohn Marino 
21950fc853eSJohn Marino 		if (MB_CUR_MAX == 1)
22050fc853eSJohn Marino 			memcpy(ret->data.cstr, s->data.cstr, (s->len));
22150fc853eSJohn Marino 		else
22250fc853eSJohn Marino 			memcpy(ret->data.wstr, s->data.wstr,
22350fc853eSJohn Marino 			    SIZEOF_WCHAR_STRING(s->len));
22450fc853eSJohn Marino 
22550fc853eSJohn Marino 		return (ret);
22650fc853eSJohn Marino 	}
22750fc853eSJohn Marino }
22850fc853eSJohn Marino 
22950fc853eSJohn Marino /*
23050fc853eSJohn Marino  * Create a new binary string from a wide character buffer.
23150fc853eSJohn Marino  */
23250fc853eSJohn Marino struct bwstring *
bwssbdup(const wchar_t * str,size_t len)23350fc853eSJohn Marino bwssbdup(const wchar_t *str, size_t len)
23450fc853eSJohn Marino {
23550fc853eSJohn Marino 
23650fc853eSJohn Marino 	if (str == NULL)
23750fc853eSJohn Marino 		return ((len == 0) ? bwsalloc(0) : NULL);
23850fc853eSJohn Marino 	else {
23950fc853eSJohn Marino 		struct bwstring *ret;
24050fc853eSJohn Marino 
24150fc853eSJohn Marino 		ret = bwsalloc(len);
24250fc853eSJohn Marino 
24350fc853eSJohn Marino 		if (MB_CUR_MAX == 1)
24450fc853eSJohn Marino 			for (size_t i = 0; i < len; ++i)
24550fc853eSJohn Marino 				ret->data.cstr[i] = (unsigned char) str[i];
24650fc853eSJohn Marino 		else
24750fc853eSJohn Marino 			memcpy(ret->data.wstr, str, SIZEOF_WCHAR_STRING(len));
24850fc853eSJohn Marino 
24950fc853eSJohn Marino 		return (ret);
25050fc853eSJohn Marino 	}
25150fc853eSJohn Marino }
25250fc853eSJohn Marino 
25350fc853eSJohn Marino /*
25450fc853eSJohn Marino  * Create a new binary string from a raw binary buffer.
25550fc853eSJohn Marino  */
25650fc853eSJohn Marino struct bwstring *
bwscsbdup(const unsigned char * str,size_t len)25750fc853eSJohn Marino bwscsbdup(const unsigned char *str, size_t len)
25850fc853eSJohn Marino {
25950fc853eSJohn Marino 	struct bwstring *ret;
26050fc853eSJohn Marino 
26150fc853eSJohn Marino 	ret = bwsalloc(len);
26250fc853eSJohn Marino 
26350fc853eSJohn Marino 	if (str) {
26450fc853eSJohn Marino 		if (MB_CUR_MAX == 1)
26550fc853eSJohn Marino 			memcpy(ret->data.cstr, str, len);
26650fc853eSJohn Marino 		else {
26750fc853eSJohn Marino 			mbstate_t mbs;
26850fc853eSJohn Marino 			const char *s;
26950fc853eSJohn Marino 			size_t charlen, chars, cptr;
27050fc853eSJohn Marino 
27150fc853eSJohn Marino 			charlen = chars = 0;
27250fc853eSJohn Marino 			cptr = 0;
27350fc853eSJohn Marino 			s = (const char *) str;
27450fc853eSJohn Marino 
27550fc853eSJohn Marino 			memset(&mbs, 0, sizeof(mbs));
27650fc853eSJohn Marino 
27750fc853eSJohn Marino 			while (cptr < len) {
27850fc853eSJohn Marino 				size_t n = MB_CUR_MAX;
27950fc853eSJohn Marino 
28050fc853eSJohn Marino 				if (n > len - cptr)
28150fc853eSJohn Marino 					n = len - cptr;
28250fc853eSJohn Marino 				charlen = mbrlen(s + cptr, n, &mbs);
28350fc853eSJohn Marino 				switch (charlen) {
28450fc853eSJohn Marino 				case 0:
28550fc853eSJohn Marino 					/* FALLTHROUGH */
28650fc853eSJohn Marino 				case (size_t) -1:
28750fc853eSJohn Marino 					/* FALLTHROUGH */
28850fc853eSJohn Marino 				case (size_t) -2:
28950fc853eSJohn Marino 					ret->data.wstr[chars++] =
29050fc853eSJohn Marino 					    (unsigned char) s[cptr];
29150fc853eSJohn Marino 					++cptr;
29250fc853eSJohn Marino 					break;
29350fc853eSJohn Marino 				default:
29450fc853eSJohn Marino 					n = mbrtowc(ret->data.wstr + (chars++),
29550fc853eSJohn Marino 					    s + cptr, charlen, &mbs);
29650fc853eSJohn Marino 					if ((n == (size_t)-1) || (n == (size_t)-2))
29750fc853eSJohn Marino 						/* NOTREACHED */
29850fc853eSJohn Marino 						err(2, "mbrtowc error");
29950fc853eSJohn Marino 					cptr += charlen;
300*94400e62SJohn Marino 				}
30150fc853eSJohn Marino 			}
30250fc853eSJohn Marino 
30350fc853eSJohn Marino 			ret->len = chars;
30450fc853eSJohn Marino 			ret->data.wstr[ret->len] = L'\0';
30550fc853eSJohn Marino 		}
30650fc853eSJohn Marino 	}
30750fc853eSJohn Marino 	return (ret);
30850fc853eSJohn Marino }
30950fc853eSJohn Marino 
31050fc853eSJohn Marino /*
31150fc853eSJohn Marino  * De-allocate object memory
31250fc853eSJohn Marino  */
31350fc853eSJohn Marino void
bwsfree(const struct bwstring * s)31450fc853eSJohn Marino bwsfree(const struct bwstring *s)
31550fc853eSJohn Marino {
31650fc853eSJohn Marino 
31750fc853eSJohn Marino 	if (s)
31850fc853eSJohn Marino 		sort_free(s);
31950fc853eSJohn Marino }
32050fc853eSJohn Marino 
32150fc853eSJohn Marino /*
32250fc853eSJohn Marino  * Copy content of src binary string to dst.
32350fc853eSJohn Marino  * If the capacity of the dst string is not sufficient,
32450fc853eSJohn Marino  * then the data is truncated.
32550fc853eSJohn Marino  */
32650fc853eSJohn Marino size_t
bwscpy(struct bwstring * dst,const struct bwstring * src)32750fc853eSJohn Marino bwscpy(struct bwstring *dst, const struct bwstring *src)
32850fc853eSJohn Marino {
32950fc853eSJohn Marino 	size_t nums = src->len;
33050fc853eSJohn Marino 
33150fc853eSJohn Marino 	if (nums > dst->len)
33250fc853eSJohn Marino 		nums = dst->len;
33350fc853eSJohn Marino 	dst->len = nums;
33450fc853eSJohn Marino 
33550fc853eSJohn Marino 	if (MB_CUR_MAX == 1) {
33650fc853eSJohn Marino 		memcpy(dst->data.cstr, src->data.cstr, nums);
33750fc853eSJohn Marino 		dst->data.cstr[dst->len] = '\0';
33850fc853eSJohn Marino 	} else {
33950fc853eSJohn Marino 		memcpy(dst->data.wstr, src->data.wstr,
34050fc853eSJohn Marino 		    SIZEOF_WCHAR_STRING(nums + 1));
34150fc853eSJohn Marino 		dst->data.wstr[dst->len] = L'\0';
34250fc853eSJohn Marino 	}
34350fc853eSJohn Marino 
34450fc853eSJohn Marino 	return (nums);
34550fc853eSJohn Marino }
34650fc853eSJohn Marino 
34750fc853eSJohn Marino /*
34850fc853eSJohn Marino  * Copy content of src binary string to dst,
34950fc853eSJohn Marino  * with specified number of symbols to be copied.
35050fc853eSJohn Marino  * If the capacity of the dst string is not sufficient,
35150fc853eSJohn Marino  * then the data is truncated.
35250fc853eSJohn Marino  */
35350fc853eSJohn Marino struct bwstring *
bwsncpy(struct bwstring * dst,const struct bwstring * src,size_t size)35450fc853eSJohn Marino bwsncpy(struct bwstring *dst, const struct bwstring *src, size_t size)
35550fc853eSJohn Marino {
35650fc853eSJohn Marino 	size_t nums = src->len;
35750fc853eSJohn Marino 
35850fc853eSJohn Marino 	if (nums > dst->len)
35950fc853eSJohn Marino 		nums = dst->len;
36050fc853eSJohn Marino 	if (nums > size)
36150fc853eSJohn Marino 		nums = size;
36250fc853eSJohn Marino 	dst->len = nums;
36350fc853eSJohn Marino 
36450fc853eSJohn Marino 	if (MB_CUR_MAX == 1) {
36550fc853eSJohn Marino 		memcpy(dst->data.cstr, src->data.cstr, nums);
36650fc853eSJohn Marino 		dst->data.cstr[dst->len] = '\0';
36750fc853eSJohn Marino 	} else {
36850fc853eSJohn Marino 		memcpy(dst->data.wstr, src->data.wstr,
36950fc853eSJohn Marino 		    SIZEOF_WCHAR_STRING(nums + 1));
37050fc853eSJohn Marino 		dst->data.wstr[dst->len] = L'\0';
37150fc853eSJohn Marino 	}
37250fc853eSJohn Marino 
37350fc853eSJohn Marino 	return (dst);
37450fc853eSJohn Marino }
37550fc853eSJohn Marino 
37650fc853eSJohn Marino /*
37750fc853eSJohn Marino  * Copy content of src binary string to dst,
37850fc853eSJohn Marino  * with specified number of symbols to be copied.
37950fc853eSJohn Marino  * An offset value can be specified, from the start of src string.
38050fc853eSJohn Marino  * If the capacity of the dst string is not sufficient,
38150fc853eSJohn Marino  * then the data is truncated.
38250fc853eSJohn Marino  */
38350fc853eSJohn Marino struct bwstring *
bwsnocpy(struct bwstring * dst,const struct bwstring * src,size_t offset,size_t size)38450fc853eSJohn Marino bwsnocpy(struct bwstring *dst, const struct bwstring *src, size_t offset,
38550fc853eSJohn Marino     size_t size)
38650fc853eSJohn Marino {
38750fc853eSJohn Marino 
38850fc853eSJohn Marino 	if (offset >= src->len) {
38950fc853eSJohn Marino 		dst->data.wstr[0] = 0;
39050fc853eSJohn Marino 		dst->len = 0;
39150fc853eSJohn Marino 	} else {
39250fc853eSJohn Marino 		size_t nums = src->len - offset;
39350fc853eSJohn Marino 
39450fc853eSJohn Marino 		if (nums > dst->len)
39550fc853eSJohn Marino 			nums = dst->len;
39650fc853eSJohn Marino 		if (nums > size)
39750fc853eSJohn Marino 			nums = size;
39850fc853eSJohn Marino 		dst->len = nums;
39950fc853eSJohn Marino 		if (MB_CUR_MAX == 1) {
40050fc853eSJohn Marino 			memcpy(dst->data.cstr, src->data.cstr + offset,
40150fc853eSJohn Marino 			    (nums));
40250fc853eSJohn Marino 			dst->data.cstr[dst->len] = '\0';
40350fc853eSJohn Marino 		} else {
40450fc853eSJohn Marino 			memcpy(dst->data.wstr, src->data.wstr + offset,
40550fc853eSJohn Marino 			    SIZEOF_WCHAR_STRING(nums));
40650fc853eSJohn Marino 			dst->data.wstr[dst->len] = L'\0';
40750fc853eSJohn Marino 		}
40850fc853eSJohn Marino 	}
40950fc853eSJohn Marino 	return (dst);
41050fc853eSJohn Marino }
41150fc853eSJohn Marino 
41250fc853eSJohn Marino /*
41350fc853eSJohn Marino  * Write binary string to the file.
41450fc853eSJohn Marino  * The output is ended either with '\n' (nl == true)
41550fc853eSJohn Marino  * or '\0' (nl == false).
41650fc853eSJohn Marino  */
41750fc853eSJohn Marino size_t
bwsfwrite(struct bwstring * bws,FILE * f,bool zero_ended)41850fc853eSJohn Marino bwsfwrite(struct bwstring *bws, FILE *f, bool zero_ended)
41950fc853eSJohn Marino {
42050fc853eSJohn Marino 
42150fc853eSJohn Marino 	if (MB_CUR_MAX == 1) {
42250fc853eSJohn Marino 		size_t len = bws->len;
42350fc853eSJohn Marino 
42450fc853eSJohn Marino 		if (!zero_ended) {
42550fc853eSJohn Marino 			bws->data.cstr[len] = '\n';
42650fc853eSJohn Marino 
42750fc853eSJohn Marino 			if (fwrite(bws->data.cstr, len + 1, 1, f) < 1)
42850fc853eSJohn Marino 				err(2, NULL);
42950fc853eSJohn Marino 
43050fc853eSJohn Marino 			bws->data.cstr[len] = '\0';
43150fc853eSJohn Marino 		} else if (fwrite(bws->data.cstr, len + 1, 1, f) < 1)
43250fc853eSJohn Marino 			err(2, NULL);
43350fc853eSJohn Marino 
43450fc853eSJohn Marino 		return (len + 1);
43550fc853eSJohn Marino 
43650fc853eSJohn Marino 	} else {
43750fc853eSJohn Marino 		wchar_t eols;
43850fc853eSJohn Marino 		size_t printed = 0;
43950fc853eSJohn Marino 
44050fc853eSJohn Marino 		eols = zero_ended ? btowc('\0') : btowc('\n');
44150fc853eSJohn Marino 
44250fc853eSJohn Marino 		while (printed < BWSLEN(bws)) {
44350fc853eSJohn Marino 			const wchar_t *s = bws->data.wstr + printed;
44450fc853eSJohn Marino 
44550fc853eSJohn Marino 			if (*s == L'\0') {
44650fc853eSJohn Marino 				int nums;
44750fc853eSJohn Marino 
44850fc853eSJohn Marino 				nums = fwprintf(f, L"%lc", *s);
44950fc853eSJohn Marino 
45050fc853eSJohn Marino 				if (nums != 1)
45150fc853eSJohn Marino 					err(2, NULL);
45250fc853eSJohn Marino 				++printed;
45350fc853eSJohn Marino 			} else {
45450fc853eSJohn Marino 				int nums;
45550fc853eSJohn Marino 
45650fc853eSJohn Marino 				nums = fwprintf(f, L"%ls", s);
45750fc853eSJohn Marino 
45850fc853eSJohn Marino 				if (nums < 1)
45950fc853eSJohn Marino 					err(2, NULL);
46050fc853eSJohn Marino 				printed += nums;
46150fc853eSJohn Marino 			}
46250fc853eSJohn Marino 		}
46350fc853eSJohn Marino 		fwprintf(f, L"%lc", eols);
46450fc853eSJohn Marino 		return (printed + 1);
46550fc853eSJohn Marino 	}
46650fc853eSJohn Marino }
46750fc853eSJohn Marino 
46850fc853eSJohn Marino /*
46950fc853eSJohn Marino  * Allocate and read a binary string from file.
47050fc853eSJohn Marino  * The strings are nl-ended or zero-ended, depending on the sort setting.
47150fc853eSJohn Marino  */
47250fc853eSJohn Marino struct bwstring *
bwsfgetln(FILE * f,size_t * len,bool zero_ended,struct reader_buffer * rb)47350fc853eSJohn Marino bwsfgetln(FILE *f, size_t *len, bool zero_ended, struct reader_buffer *rb)
47450fc853eSJohn Marino {
47550fc853eSJohn Marino 	wint_t eols;
47650fc853eSJohn Marino 
47750fc853eSJohn Marino 	eols = zero_ended ? btowc('\0') : btowc('\n');
47850fc853eSJohn Marino 
47950fc853eSJohn Marino 	if (!zero_ended && (MB_CUR_MAX > 1)) {
48050fc853eSJohn Marino 		wchar_t *ret;
48150fc853eSJohn Marino 
48250fc853eSJohn Marino 		ret = fgetwln(f, len);
48350fc853eSJohn Marino 
48450fc853eSJohn Marino 		if (ret == NULL) {
48550fc853eSJohn Marino 			if (!feof(f))
48650fc853eSJohn Marino 				err(2, NULL);
48750fc853eSJohn Marino 			return (NULL);
48850fc853eSJohn Marino 		}
48950fc853eSJohn Marino 		if (*len > 0) {
49050fc853eSJohn Marino 			if (ret[*len - 1] == (wchar_t)eols)
49150fc853eSJohn Marino 				--(*len);
49250fc853eSJohn Marino 		}
49350fc853eSJohn Marino 		return (bwssbdup(ret, *len));
49450fc853eSJohn Marino 
49550fc853eSJohn Marino 	} else if (!zero_ended && (MB_CUR_MAX == 1)) {
49650fc853eSJohn Marino 		char *ret;
49750fc853eSJohn Marino 
49850fc853eSJohn Marino 		ret = fgetln(f, len);
49950fc853eSJohn Marino 
50050fc853eSJohn Marino 		if (ret == NULL) {
50150fc853eSJohn Marino 			if (!feof(f))
50250fc853eSJohn Marino 				err(2, NULL);
50350fc853eSJohn Marino 			return (NULL);
50450fc853eSJohn Marino 		}
50550fc853eSJohn Marino 		if (*len > 0) {
50650fc853eSJohn Marino 			if (ret[*len - 1] == '\n')
50750fc853eSJohn Marino 				--(*len);
50850fc853eSJohn Marino 		}
50950fc853eSJohn Marino 		return (bwscsbdup((unsigned char*)ret, *len));
51050fc853eSJohn Marino 
51150fc853eSJohn Marino 	} else {
51250fc853eSJohn Marino 		*len = 0;
51350fc853eSJohn Marino 
51450fc853eSJohn Marino 		if (feof(f))
51550fc853eSJohn Marino 			return (NULL);
51650fc853eSJohn Marino 
51750fc853eSJohn Marino 		if (2 >= rb->fgetwln_z_buffer_size) {
51850fc853eSJohn Marino 			rb->fgetwln_z_buffer_size += 256;
51950fc853eSJohn Marino 			rb->fgetwln_z_buffer = sort_realloc(rb->fgetwln_z_buffer,
52050fc853eSJohn Marino 			    sizeof(wchar_t) * rb->fgetwln_z_buffer_size);
52150fc853eSJohn Marino 		}
52250fc853eSJohn Marino 		rb->fgetwln_z_buffer[*len] = 0;
52350fc853eSJohn Marino 
52450fc853eSJohn Marino 		if (MB_CUR_MAX == 1)
52550fc853eSJohn Marino 			while (!feof(f)) {
52650fc853eSJohn Marino 				int c;
52750fc853eSJohn Marino 
52850fc853eSJohn Marino 				c = fgetc(f);
52950fc853eSJohn Marino 
53050fc853eSJohn Marino 				if (c == EOF) {
53150fc853eSJohn Marino 					if (*len == 0)
53250fc853eSJohn Marino 						return (NULL);
53350fc853eSJohn Marino 					goto line_read_done;
53450fc853eSJohn Marino 				}
53550fc853eSJohn Marino 				if (c == eols)
53650fc853eSJohn Marino 					goto line_read_done;
53750fc853eSJohn Marino 
53850fc853eSJohn Marino 				if (*len + 1 >= rb->fgetwln_z_buffer_size) {
53950fc853eSJohn Marino 					rb->fgetwln_z_buffer_size += 256;
54050fc853eSJohn Marino 					rb->fgetwln_z_buffer = sort_realloc(rb->fgetwln_z_buffer,
54150fc853eSJohn Marino 					    SIZEOF_WCHAR_STRING(rb->fgetwln_z_buffer_size));
54250fc853eSJohn Marino 				}
54350fc853eSJohn Marino 
54450fc853eSJohn Marino 				rb->fgetwln_z_buffer[*len] = c;
54550fc853eSJohn Marino 				rb->fgetwln_z_buffer[++(*len)] = 0;
54650fc853eSJohn Marino 			}
54750fc853eSJohn Marino 		else
54850fc853eSJohn Marino 			while (!feof(f)) {
54950fc853eSJohn Marino 				wint_t c = 0;
55050fc853eSJohn Marino 
55150fc853eSJohn Marino 				c = fgetwc(f);
55250fc853eSJohn Marino 
55350fc853eSJohn Marino 				if (c == WEOF) {
55450fc853eSJohn Marino 					if (*len == 0)
55550fc853eSJohn Marino 						return (NULL);
55650fc853eSJohn Marino 					goto line_read_done;
55750fc853eSJohn Marino 				}
55850fc853eSJohn Marino 				if (c == eols)
55950fc853eSJohn Marino 					goto line_read_done;
56050fc853eSJohn Marino 
56150fc853eSJohn Marino 				if (*len + 1 >= rb->fgetwln_z_buffer_size) {
56250fc853eSJohn Marino 					rb->fgetwln_z_buffer_size += 256;
56350fc853eSJohn Marino 					rb->fgetwln_z_buffer = sort_realloc(rb->fgetwln_z_buffer,
56450fc853eSJohn Marino 					    SIZEOF_WCHAR_STRING(rb->fgetwln_z_buffer_size));
56550fc853eSJohn Marino 				}
56650fc853eSJohn Marino 
56750fc853eSJohn Marino 				rb->fgetwln_z_buffer[*len] = c;
56850fc853eSJohn Marino 				rb->fgetwln_z_buffer[++(*len)] = 0;
56950fc853eSJohn Marino 			}
57050fc853eSJohn Marino 
57150fc853eSJohn Marino line_read_done:
57250fc853eSJohn Marino 		/* we do not count the last 0 */
57350fc853eSJohn Marino 		return (bwssbdup(rb->fgetwln_z_buffer, *len));
57450fc853eSJohn Marino 	}
57550fc853eSJohn Marino }
57650fc853eSJohn Marino 
57750fc853eSJohn Marino int
bwsncmp(const struct bwstring * bws1,const struct bwstring * bws2,size_t offset,size_t len)57850fc853eSJohn Marino bwsncmp(const struct bwstring *bws1, const struct bwstring *bws2,
57950fc853eSJohn Marino     size_t offset, size_t len)
58050fc853eSJohn Marino {
58150fc853eSJohn Marino 	size_t cmp_len, len1, len2;
58250fc853eSJohn Marino 	int res = 0;
58350fc853eSJohn Marino 
58450fc853eSJohn Marino 	cmp_len = 0;
58550fc853eSJohn Marino 	len1 = bws1->len;
58650fc853eSJohn Marino 	len2 = bws2->len;
58750fc853eSJohn Marino 
58850fc853eSJohn Marino 	if (len1 <= offset) {
58950fc853eSJohn Marino 		return ((len2 <= offset) ? 0 : -1);
59050fc853eSJohn Marino 	} else {
59150fc853eSJohn Marino 		if (len2 <= offset)
59250fc853eSJohn Marino 			return (+1);
59350fc853eSJohn Marino 		else {
59450fc853eSJohn Marino 			len1 -= offset;
59550fc853eSJohn Marino 			len2 -= offset;
59650fc853eSJohn Marino 
59750fc853eSJohn Marino 			cmp_len = len1;
59850fc853eSJohn Marino 
59950fc853eSJohn Marino 			if (len2 < cmp_len)
60050fc853eSJohn Marino 				cmp_len = len2;
60150fc853eSJohn Marino 
60250fc853eSJohn Marino 			if (len < cmp_len)
60350fc853eSJohn Marino 				cmp_len = len;
60450fc853eSJohn Marino 
60550fc853eSJohn Marino 			if (MB_CUR_MAX == 1) {
60650fc853eSJohn Marino 				const unsigned char *s1, *s2;
60750fc853eSJohn Marino 
60850fc853eSJohn Marino 				s1 = bws1->data.cstr + offset;
60950fc853eSJohn Marino 				s2 = bws2->data.cstr + offset;
61050fc853eSJohn Marino 
61150fc853eSJohn Marino 				res = memcmp(s1, s2, cmp_len);
61250fc853eSJohn Marino 
61350fc853eSJohn Marino 			} else {
61450fc853eSJohn Marino 				const wchar_t *s1, *s2;
61550fc853eSJohn Marino 
61650fc853eSJohn Marino 				s1 = bws1->data.wstr + offset;
61750fc853eSJohn Marino 				s2 = bws2->data.wstr + offset;
61850fc853eSJohn Marino 
61950fc853eSJohn Marino 				res = memcmp(s1, s2, SIZEOF_WCHAR_STRING(cmp_len));
62050fc853eSJohn Marino 			}
62150fc853eSJohn Marino 		}
62250fc853eSJohn Marino 	}
62350fc853eSJohn Marino 
62450fc853eSJohn Marino 	if (res == 0) {
62550fc853eSJohn Marino 		if (len1 < cmp_len && len1 < len2)
62650fc853eSJohn Marino 			res = -1;
62750fc853eSJohn Marino 		else if (len2 < cmp_len && len2 < len1)
62850fc853eSJohn Marino 			res = +1;
62950fc853eSJohn Marino 	}
63050fc853eSJohn Marino 
63150fc853eSJohn Marino 	return (res);
63250fc853eSJohn Marino }
63350fc853eSJohn Marino 
63450fc853eSJohn Marino int
bwscmp(const struct bwstring * bws1,const struct bwstring * bws2,size_t offset)63550fc853eSJohn Marino bwscmp(const struct bwstring *bws1, const struct bwstring *bws2, size_t offset)
63650fc853eSJohn Marino {
63750fc853eSJohn Marino 	size_t len1, len2, cmp_len;
63850fc853eSJohn Marino 	int res;
63950fc853eSJohn Marino 
64050fc853eSJohn Marino 	len1 = bws1->len;
64150fc853eSJohn Marino 	len2 = bws2->len;
64250fc853eSJohn Marino 
64350fc853eSJohn Marino 	len1 -= offset;
64450fc853eSJohn Marino 	len2 -= offset;
64550fc853eSJohn Marino 
64650fc853eSJohn Marino 	cmp_len = len1;
64750fc853eSJohn Marino 
64850fc853eSJohn Marino 	if (len2 < cmp_len)
64950fc853eSJohn Marino 		cmp_len = len2;
65050fc853eSJohn Marino 
65150fc853eSJohn Marino 	res = bwsncmp(bws1, bws2, offset, cmp_len);
65250fc853eSJohn Marino 
65350fc853eSJohn Marino 	if (res == 0) {
65450fc853eSJohn Marino 		if( len1 < len2)
65550fc853eSJohn Marino 			res = -1;
65650fc853eSJohn Marino 		else if (len2 < len1)
65750fc853eSJohn Marino 			res = +1;
65850fc853eSJohn Marino 	}
65950fc853eSJohn Marino 
66050fc853eSJohn Marino 	return (res);
66150fc853eSJohn Marino }
66250fc853eSJohn Marino 
66350fc853eSJohn Marino int
bws_iterator_cmp(bwstring_iterator iter1,bwstring_iterator iter2,size_t len)66450fc853eSJohn Marino bws_iterator_cmp(bwstring_iterator iter1, bwstring_iterator iter2, size_t len)
66550fc853eSJohn Marino {
66650fc853eSJohn Marino 	wchar_t c1, c2;
66750fc853eSJohn Marino 	size_t i = 0;
66850fc853eSJohn Marino 
66950fc853eSJohn Marino 	for (i = 0; i < len; ++i) {
67050fc853eSJohn Marino 		c1 = bws_get_iter_value(iter1);
67150fc853eSJohn Marino 		c2 = bws_get_iter_value(iter2);
67250fc853eSJohn Marino 		if (c1 != c2)
67350fc853eSJohn Marino 			return (c1 - c2);
67450fc853eSJohn Marino 		iter1 = bws_iterator_inc(iter1, 1);
67550fc853eSJohn Marino 		iter2 = bws_iterator_inc(iter2, 1);
67650fc853eSJohn Marino 	}
67750fc853eSJohn Marino 
67850fc853eSJohn Marino 	return (0);
67950fc853eSJohn Marino }
68050fc853eSJohn Marino 
68150fc853eSJohn Marino int
bwscoll(const struct bwstring * bws1,const struct bwstring * bws2,size_t offset)68250fc853eSJohn Marino bwscoll(const struct bwstring *bws1, const struct bwstring *bws2, size_t offset)
68350fc853eSJohn Marino {
68450fc853eSJohn Marino 	size_t len1, len2;
68550fc853eSJohn Marino 
68650fc853eSJohn Marino 	len1 = bws1->len;
68750fc853eSJohn Marino 	len2 = bws2->len;
68850fc853eSJohn Marino 
68950fc853eSJohn Marino 	if (len1 <= offset)
69050fc853eSJohn Marino 		return ((len2 <= offset) ? 0 : -1);
69150fc853eSJohn Marino 	else {
69250fc853eSJohn Marino 		if (len2 <= offset)
69350fc853eSJohn Marino 			return (+1);
69450fc853eSJohn Marino 		else {
69550fc853eSJohn Marino 			len1 -= offset;
69650fc853eSJohn Marino 			len2 -= offset;
69750fc853eSJohn Marino 
69850fc853eSJohn Marino 			if (MB_CUR_MAX == 1) {
69950fc853eSJohn Marino 				const unsigned char *s1, *s2;
70050fc853eSJohn Marino 
70150fc853eSJohn Marino 				s1 = bws1->data.cstr + offset;
70250fc853eSJohn Marino 				s2 = bws2->data.cstr + offset;
70350fc853eSJohn Marino 
70450fc853eSJohn Marino 				if (byte_sort) {
70550fc853eSJohn Marino 					int res = 0;
70650fc853eSJohn Marino 
70750fc853eSJohn Marino 					if (len1 > len2) {
70850fc853eSJohn Marino 						res = memcmp(s1, s2, len2);
70950fc853eSJohn Marino 						if (!res)
71050fc853eSJohn Marino 							res = +1;
71150fc853eSJohn Marino 					} else if (len1 < len2) {
71250fc853eSJohn Marino 						res = memcmp(s1, s2, len1);
71350fc853eSJohn Marino 						if (!res)
71450fc853eSJohn Marino 							res = -1;
71550fc853eSJohn Marino 					} else
71650fc853eSJohn Marino 						res = memcmp(s1, s2, len1);
71750fc853eSJohn Marino 
71850fc853eSJohn Marino 					return (res);
71950fc853eSJohn Marino 
72050fc853eSJohn Marino 				} else {
72150fc853eSJohn Marino 					int res = 0;
72250fc853eSJohn Marino 					size_t i, maxlen;
72350fc853eSJohn Marino 
72450fc853eSJohn Marino 					i = 0;
72550fc853eSJohn Marino 					maxlen = len1;
72650fc853eSJohn Marino 
72750fc853eSJohn Marino 					if (maxlen > len2)
72850fc853eSJohn Marino 						maxlen = len2;
72950fc853eSJohn Marino 
73050fc853eSJohn Marino 					while (i < maxlen) {
73150fc853eSJohn Marino 						/* goto next non-zero part: */
73250fc853eSJohn Marino 						while ((i < maxlen) &&
73350fc853eSJohn Marino 						    !s1[i] && !s2[i])
73450fc853eSJohn Marino 							++i;
73550fc853eSJohn Marino 
73650fc853eSJohn Marino 						if (i >= maxlen)
73750fc853eSJohn Marino 							break;
73850fc853eSJohn Marino 
73950fc853eSJohn Marino 						if (s1[i] == 0) {
74050fc853eSJohn Marino 							if (s2[i] == 0)
74150fc853eSJohn Marino 								/* NOTREACHED */
74250fc853eSJohn Marino 								err(2, "bwscoll error 01");
74350fc853eSJohn Marino 							else
74450fc853eSJohn Marino 								return (-1);
74550fc853eSJohn Marino 						} else if (s2[i] == 0)
74650fc853eSJohn Marino 							return (+1);
74750fc853eSJohn Marino 
74850fc853eSJohn Marino 						res = strcoll((const char*)(s1 + i), (const char*)(s2 + i));
74950fc853eSJohn Marino 						if (res)
75050fc853eSJohn Marino 							return (res);
75150fc853eSJohn Marino 
75250fc853eSJohn Marino 						while ((i < maxlen) &&
75350fc853eSJohn Marino 						    s1[i] && s2[i])
75450fc853eSJohn Marino 							++i;
75550fc853eSJohn Marino 
75650fc853eSJohn Marino 						if (i >= maxlen)
75750fc853eSJohn Marino 							break;
75850fc853eSJohn Marino 
75950fc853eSJohn Marino 						if (s1[i] == 0) {
76050fc853eSJohn Marino 							if (s2[i] == 0) {
76150fc853eSJohn Marino 								++i;
76250fc853eSJohn Marino 								continue;
76350fc853eSJohn Marino 							} else
76450fc853eSJohn Marino 								return (-1);
76550fc853eSJohn Marino 						} else if (s2[i] == 0)
76650fc853eSJohn Marino 							return (+1);
76750fc853eSJohn Marino 						else
76850fc853eSJohn Marino 							/* NOTREACHED */
76950fc853eSJohn Marino 							err(2, "bwscoll error 02");
77050fc853eSJohn Marino 					}
77150fc853eSJohn Marino 
77250fc853eSJohn Marino 					if (len1 < len2)
77350fc853eSJohn Marino 						return (-1);
77450fc853eSJohn Marino 					else if (len1 > len2)
77550fc853eSJohn Marino 						return (+1);
77650fc853eSJohn Marino 
77750fc853eSJohn Marino 					return (0);
77850fc853eSJohn Marino 				}
77950fc853eSJohn Marino 			} else {
78050fc853eSJohn Marino 				const wchar_t *s1, *s2;
78150fc853eSJohn Marino 				size_t i, maxlen;
78250fc853eSJohn Marino 				int res = 0;
78350fc853eSJohn Marino 
78450fc853eSJohn Marino 				s1 = bws1->data.wstr + offset;
78550fc853eSJohn Marino 				s2 = bws2->data.wstr + offset;
78650fc853eSJohn Marino 
78750fc853eSJohn Marino 				i = 0;
78850fc853eSJohn Marino 				maxlen = len1;
78950fc853eSJohn Marino 
79050fc853eSJohn Marino 				if (maxlen > len2)
79150fc853eSJohn Marino 					maxlen = len2;
79250fc853eSJohn Marino 
79350fc853eSJohn Marino 				while (i < maxlen) {
79450fc853eSJohn Marino 
79550fc853eSJohn Marino 					/* goto next non-zero part: */
79650fc853eSJohn Marino 					while ((i < maxlen) &&
79750fc853eSJohn Marino 					    !s1[i] && !s2[i])
79850fc853eSJohn Marino 						++i;
79950fc853eSJohn Marino 
80050fc853eSJohn Marino 					if (i >= maxlen)
80150fc853eSJohn Marino 						break;
80250fc853eSJohn Marino 
80350fc853eSJohn Marino 					if (s1[i] == 0) {
80450fc853eSJohn Marino 						if (s2[i] == 0)
80550fc853eSJohn Marino 							/* NOTREACHED */
80650fc853eSJohn Marino 							err(2, "bwscoll error 1");
80750fc853eSJohn Marino 						else
80850fc853eSJohn Marino 							return (-1);
80950fc853eSJohn Marino 					} else if (s2[i] == 0)
81050fc853eSJohn Marino 						return (+1);
81150fc853eSJohn Marino 
81250fc853eSJohn Marino 					res = wide_str_coll(s1 + i, s2 + i);
81350fc853eSJohn Marino 					if (res)
81450fc853eSJohn Marino 						return (res);
81550fc853eSJohn Marino 
81650fc853eSJohn Marino 					while ((i < maxlen) && s1[i] && s2[i])
81750fc853eSJohn Marino 						++i;
81850fc853eSJohn Marino 
81950fc853eSJohn Marino 					if (i >= maxlen)
82050fc853eSJohn Marino 						break;
82150fc853eSJohn Marino 
82250fc853eSJohn Marino 					if (s1[i] == 0) {
82350fc853eSJohn Marino 						if (s2[i] == 0) {
82450fc853eSJohn Marino 							++i;
82550fc853eSJohn Marino 							continue;
82650fc853eSJohn Marino 						} else
82750fc853eSJohn Marino 							return (-1);
82850fc853eSJohn Marino 					} else if (s2[i] == 0)
82950fc853eSJohn Marino 						return (+1);
83050fc853eSJohn Marino 					else
83150fc853eSJohn Marino 						/* NOTREACHED */
83250fc853eSJohn Marino 						err(2, "bwscoll error 2");
83350fc853eSJohn Marino 				}
83450fc853eSJohn Marino 
83550fc853eSJohn Marino 				if (len1 < len2)
83650fc853eSJohn Marino 					return (-1);
83750fc853eSJohn Marino 				else if (len1 > len2)
83850fc853eSJohn Marino 					return (+1);
83950fc853eSJohn Marino 
84050fc853eSJohn Marino 				return (0);
84150fc853eSJohn Marino 			}
84250fc853eSJohn Marino 		}
84350fc853eSJohn Marino 	}
84450fc853eSJohn Marino }
84550fc853eSJohn Marino 
84650fc853eSJohn Marino /*
84750fc853eSJohn Marino  * Correction of the system API
84850fc853eSJohn Marino  */
84950fc853eSJohn Marino double
bwstod(struct bwstring * s0,bool * empty)85050fc853eSJohn Marino bwstod(struct bwstring *s0, bool *empty)
85150fc853eSJohn Marino {
85250fc853eSJohn Marino 	double ret = 0;
85350fc853eSJohn Marino 
85450fc853eSJohn Marino 	if (MB_CUR_MAX == 1) {
85550fc853eSJohn Marino 		unsigned char *end, *s;
85650fc853eSJohn Marino 		char *ep;
85750fc853eSJohn Marino 
85850fc853eSJohn Marino 		s = s0->data.cstr;
85950fc853eSJohn Marino 		end = s + s0->len;
86050fc853eSJohn Marino 		ep = NULL;
86150fc853eSJohn Marino 
86250fc853eSJohn Marino 		while (isblank(*s) && s < end)
86350fc853eSJohn Marino 			++s;
86450fc853eSJohn Marino 
86550fc853eSJohn Marino 		if (!isprint(*s)) {
86650fc853eSJohn Marino 			*empty = true;
86750fc853eSJohn Marino 			return (0);
86850fc853eSJohn Marino 		}
86950fc853eSJohn Marino 
87050fc853eSJohn Marino 		ret = strtod((char*)s, &ep);
87150fc853eSJohn Marino 		if ((unsigned char*) ep == s) {
87250fc853eSJohn Marino 			*empty = true;
87350fc853eSJohn Marino 			return (0);
87450fc853eSJohn Marino 		}
87550fc853eSJohn Marino 	} else {
87650fc853eSJohn Marino 		wchar_t *end, *ep, *s;
87750fc853eSJohn Marino 
87850fc853eSJohn Marino 		s = s0->data.wstr;
87950fc853eSJohn Marino 		end = s + s0->len;
88050fc853eSJohn Marino 		ep = NULL;
88150fc853eSJohn Marino 
88250fc853eSJohn Marino 		while (iswblank(*s) && s < end)
88350fc853eSJohn Marino 			++s;
88450fc853eSJohn Marino 
88550fc853eSJohn Marino 		if (!iswprint(*s)) {
88650fc853eSJohn Marino 			*empty = true;
88750fc853eSJohn Marino 			return (0);
88850fc853eSJohn Marino 		}
88950fc853eSJohn Marino 
89050fc853eSJohn Marino 		ret = wcstod(s, &ep);
89150fc853eSJohn Marino 		if (ep == s) {
89250fc853eSJohn Marino 			*empty = true;
89350fc853eSJohn Marino 			return (0);
89450fc853eSJohn Marino 		}
89550fc853eSJohn Marino 	}
89650fc853eSJohn Marino 
89750fc853eSJohn Marino 	*empty = false;
89850fc853eSJohn Marino 	return (ret);
89950fc853eSJohn Marino }
90050fc853eSJohn Marino 
90150fc853eSJohn Marino /*
90250fc853eSJohn Marino  * A helper function for monthcoll.  If a line matches
90350fc853eSJohn Marino  * a month name, it returns (number of the month - 1),
90450fc853eSJohn Marino  * while if there is no match, it just return -1.
90550fc853eSJohn Marino  */
90650fc853eSJohn Marino 
90750fc853eSJohn Marino int
bws_month_score(const struct bwstring * s0)90850fc853eSJohn Marino bws_month_score(const struct bwstring *s0)
90950fc853eSJohn Marino {
91050fc853eSJohn Marino 
91150fc853eSJohn Marino 	if (MB_CUR_MAX == 1) {
91250fc853eSJohn Marino 		const unsigned char *end, *s;
91350fc853eSJohn Marino 
91450fc853eSJohn Marino 		s = s0->data.cstr;
91550fc853eSJohn Marino 		end = s + s0->len;
91650fc853eSJohn Marino 
91750fc853eSJohn Marino 		while (isblank(*s) && s < end)
91850fc853eSJohn Marino 			++s;
91950fc853eSJohn Marino 
92050fc853eSJohn Marino 		for (int i = 11; i >= 0; --i) {
92150fc853eSJohn Marino 			if (cmonths[i] &&
92250fc853eSJohn Marino 			    (s == (unsigned char*)strstr((const char*)s, (char*)(cmonths[i]))))
92350fc853eSJohn Marino 				return (i);
92450fc853eSJohn Marino 		}
92550fc853eSJohn Marino 
92650fc853eSJohn Marino 	} else {
92750fc853eSJohn Marino 		const wchar_t *end, *s;
92850fc853eSJohn Marino 
92950fc853eSJohn Marino 		s = s0->data.wstr;
93050fc853eSJohn Marino 		end = s + s0->len;
93150fc853eSJohn Marino 
93250fc853eSJohn Marino 		while (iswblank(*s) && s < end)
93350fc853eSJohn Marino 			++s;
93450fc853eSJohn Marino 
93550fc853eSJohn Marino 		for (int i = 11; i >= 0; --i) {
93650fc853eSJohn Marino 			if (wmonths[i] && (s == wcsstr(s, wmonths[i])))
93750fc853eSJohn Marino 				return (i);
93850fc853eSJohn Marino 		}
93950fc853eSJohn Marino 	}
94050fc853eSJohn Marino 
94150fc853eSJohn Marino 	return (-1);
94250fc853eSJohn Marino }
94350fc853eSJohn Marino 
94450fc853eSJohn Marino /*
94550fc853eSJohn Marino  * Rips out leading blanks (-b).
94650fc853eSJohn Marino  */
94750fc853eSJohn Marino struct bwstring *
ignore_leading_blanks(struct bwstring * str)94850fc853eSJohn Marino ignore_leading_blanks(struct bwstring *str)
94950fc853eSJohn Marino {
95050fc853eSJohn Marino 
95150fc853eSJohn Marino 	if (MB_CUR_MAX == 1) {
95250fc853eSJohn Marino 		unsigned char *dst, *end, *src;
95350fc853eSJohn Marino 
95450fc853eSJohn Marino 		src = str->data.cstr;
95550fc853eSJohn Marino 		dst = src;
95650fc853eSJohn Marino 		end = src + str->len;
95750fc853eSJohn Marino 
95850fc853eSJohn Marino 		while (src < end && isblank(*src))
95950fc853eSJohn Marino 			++src;
96050fc853eSJohn Marino 
96150fc853eSJohn Marino 		if (src != dst) {
96250fc853eSJohn Marino 			size_t newlen;
96350fc853eSJohn Marino 
96450fc853eSJohn Marino 			newlen = BWSLEN(str) - (src - dst);
96550fc853eSJohn Marino 
96650fc853eSJohn Marino 			while (src < end) {
96750fc853eSJohn Marino 				*dst = *src;
96850fc853eSJohn Marino 				++dst;
96950fc853eSJohn Marino 				++src;
97050fc853eSJohn Marino 			}
97150fc853eSJohn Marino 			bws_setlen(str, newlen);
97250fc853eSJohn Marino 		}
97350fc853eSJohn Marino 	} else {
97450fc853eSJohn Marino 		wchar_t *dst, *end, *src;
97550fc853eSJohn Marino 
97650fc853eSJohn Marino 		src = str->data.wstr;
97750fc853eSJohn Marino 		dst = src;
97850fc853eSJohn Marino 		end = src + str->len;
97950fc853eSJohn Marino 
98050fc853eSJohn Marino 		while (src < end && iswblank(*src))
98150fc853eSJohn Marino 			++src;
98250fc853eSJohn Marino 
98350fc853eSJohn Marino 		if (src != dst) {
98450fc853eSJohn Marino 
98550fc853eSJohn Marino 			size_t newlen = BWSLEN(str) - (src - dst);
98650fc853eSJohn Marino 
98750fc853eSJohn Marino 			while (src < end) {
98850fc853eSJohn Marino 				*dst = *src;
98950fc853eSJohn Marino 				++dst;
99050fc853eSJohn Marino 				++src;
99150fc853eSJohn Marino 			}
99250fc853eSJohn Marino 			bws_setlen(str, newlen);
99350fc853eSJohn Marino 
99450fc853eSJohn Marino 		}
99550fc853eSJohn Marino 	}
99650fc853eSJohn Marino 	return (str);
99750fc853eSJohn Marino }
99850fc853eSJohn Marino 
99950fc853eSJohn Marino /*
100050fc853eSJohn Marino  * Rips out nonprinting characters (-i).
100150fc853eSJohn Marino  */
100250fc853eSJohn Marino struct bwstring *
ignore_nonprinting(struct bwstring * str)100350fc853eSJohn Marino ignore_nonprinting(struct bwstring *str)
100450fc853eSJohn Marino {
100550fc853eSJohn Marino 	size_t newlen = str->len;
100650fc853eSJohn Marino 
100750fc853eSJohn Marino 	if (MB_CUR_MAX == 1) {
100850fc853eSJohn Marino 		unsigned char *dst, *end, *src;
100950fc853eSJohn Marino 		unsigned char c;
101050fc853eSJohn Marino 
101150fc853eSJohn Marino 		src = str->data.cstr;
101250fc853eSJohn Marino 		dst = src;
101350fc853eSJohn Marino 		end = src + str->len;
101450fc853eSJohn Marino 
101550fc853eSJohn Marino 		while (src < end) {
101650fc853eSJohn Marino 			c = *src;
101750fc853eSJohn Marino 			if (isprint(c)) {
101850fc853eSJohn Marino 				*dst = c;
101950fc853eSJohn Marino 				++dst;
102050fc853eSJohn Marino 				++src;
102150fc853eSJohn Marino 			} else {
102250fc853eSJohn Marino 				++src;
102350fc853eSJohn Marino 				--newlen;
102450fc853eSJohn Marino 			}
102550fc853eSJohn Marino 		}
102650fc853eSJohn Marino 	} else {
102750fc853eSJohn Marino 		wchar_t *dst, *end, *src;
102850fc853eSJohn Marino 		wchar_t c;
102950fc853eSJohn Marino 
103050fc853eSJohn Marino 		src = str->data.wstr;
103150fc853eSJohn Marino 		dst = src;
103250fc853eSJohn Marino 		end = src + str->len;
103350fc853eSJohn Marino 
103450fc853eSJohn Marino 		while (src < end) {
103550fc853eSJohn Marino 			c = *src;
103650fc853eSJohn Marino 			if (iswprint(c)) {
103750fc853eSJohn Marino 				*dst = c;
103850fc853eSJohn Marino 				++dst;
103950fc853eSJohn Marino 				++src;
104050fc853eSJohn Marino 			} else {
104150fc853eSJohn Marino 				++src;
104250fc853eSJohn Marino 				--newlen;
104350fc853eSJohn Marino 			}
104450fc853eSJohn Marino 		}
104550fc853eSJohn Marino 	}
104650fc853eSJohn Marino 	bws_setlen(str, newlen);
104750fc853eSJohn Marino 
104850fc853eSJohn Marino 	return (str);
104950fc853eSJohn Marino }
105050fc853eSJohn Marino 
105150fc853eSJohn Marino /*
105250fc853eSJohn Marino  * Rips out any characters that are not alphanumeric characters
105350fc853eSJohn Marino  * nor blanks (-d).
105450fc853eSJohn Marino  */
105550fc853eSJohn Marino struct bwstring *
dictionary_order(struct bwstring * str)105650fc853eSJohn Marino dictionary_order(struct bwstring *str)
105750fc853eSJohn Marino {
105850fc853eSJohn Marino 	size_t newlen = str->len;
105950fc853eSJohn Marino 
106050fc853eSJohn Marino 	if (MB_CUR_MAX == 1) {
106150fc853eSJohn Marino 		unsigned char *dst, *end, *src;
106250fc853eSJohn Marino 		unsigned char c;
106350fc853eSJohn Marino 
106450fc853eSJohn Marino 		src = str->data.cstr;
106550fc853eSJohn Marino 		dst = src;
106650fc853eSJohn Marino 		end = src + str->len;
106750fc853eSJohn Marino 
106850fc853eSJohn Marino 		while (src < end) {
106950fc853eSJohn Marino 			c = *src;
107050fc853eSJohn Marino 			if (isalnum(c) || isblank(c)) {
107150fc853eSJohn Marino 				*dst = c;
107250fc853eSJohn Marino 				++dst;
107350fc853eSJohn Marino 				++src;
107450fc853eSJohn Marino 			} else {
107550fc853eSJohn Marino 				++src;
107650fc853eSJohn Marino 				--newlen;
107750fc853eSJohn Marino 			}
107850fc853eSJohn Marino 		}
107950fc853eSJohn Marino 	} else {
108050fc853eSJohn Marino 		wchar_t *dst, *end, *src;
108150fc853eSJohn Marino 		wchar_t c;
108250fc853eSJohn Marino 
108350fc853eSJohn Marino 		src = str->data.wstr;
108450fc853eSJohn Marino 		dst = src;
108550fc853eSJohn Marino 		end = src + str->len;
108650fc853eSJohn Marino 
108750fc853eSJohn Marino 		while (src < end) {
108850fc853eSJohn Marino 			c = *src;
108950fc853eSJohn Marino 			if (iswalnum(c) || iswblank(c)) {
109050fc853eSJohn Marino 				*dst = c;
109150fc853eSJohn Marino 				++dst;
109250fc853eSJohn Marino 				++src;
109350fc853eSJohn Marino 			} else {
109450fc853eSJohn Marino 				++src;
109550fc853eSJohn Marino 				--newlen;
109650fc853eSJohn Marino 			}
109750fc853eSJohn Marino 		}
109850fc853eSJohn Marino 	}
109950fc853eSJohn Marino 	bws_setlen(str, newlen);
110050fc853eSJohn Marino 
110150fc853eSJohn Marino 	return (str);
110250fc853eSJohn Marino }
110350fc853eSJohn Marino 
110450fc853eSJohn Marino /*
110550fc853eSJohn Marino  * Converts string to lower case(-f).
110650fc853eSJohn Marino  */
110750fc853eSJohn Marino struct bwstring *
ignore_case(struct bwstring * str)110850fc853eSJohn Marino ignore_case(struct bwstring *str)
110950fc853eSJohn Marino {
111050fc853eSJohn Marino 
111150fc853eSJohn Marino 	if (MB_CUR_MAX == 1) {
111250fc853eSJohn Marino 		unsigned char *end, *s;
111350fc853eSJohn Marino 
111450fc853eSJohn Marino 		s = str->data.cstr;
111550fc853eSJohn Marino 		end = s + str->len;
111650fc853eSJohn Marino 
111750fc853eSJohn Marino 		while (s < end) {
111850fc853eSJohn Marino 			*s = toupper(*s);
111950fc853eSJohn Marino 			++s;
112050fc853eSJohn Marino 		}
112150fc853eSJohn Marino 	} else {
112250fc853eSJohn Marino 		wchar_t *end, *s;
112350fc853eSJohn Marino 
112450fc853eSJohn Marino 		s = str->data.wstr;
112550fc853eSJohn Marino 		end = s + str->len;
112650fc853eSJohn Marino 
112750fc853eSJohn Marino 		while (s < end) {
112850fc853eSJohn Marino 			*s = towupper(*s);
112950fc853eSJohn Marino 			++s;
113050fc853eSJohn Marino 		}
113150fc853eSJohn Marino 	}
113250fc853eSJohn Marino 	return (str);
113350fc853eSJohn Marino }
113450fc853eSJohn Marino 
113550fc853eSJohn Marino void
bws_disorder_warnx(struct bwstring * s,const char * fn,size_t pos)113650fc853eSJohn Marino bws_disorder_warnx(struct bwstring *s, const char *fn, size_t pos)
113750fc853eSJohn Marino {
113850fc853eSJohn Marino 
113950fc853eSJohn Marino 	if (MB_CUR_MAX == 1)
114050fc853eSJohn Marino 		warnx("%s:%zu: disorder: %s", fn, pos + 1, s->data.cstr);
114150fc853eSJohn Marino 	else
114250fc853eSJohn Marino 		warnx("%s:%zu: disorder: %ls", fn, pos + 1, s->data.wstr);
114350fc853eSJohn Marino }
1144