1 /*- 2 * Copyright (C) 2009 Gabor Kovesdan <gabor@FreeBSD.org> 3 * Copyright (C) 2012 Oleg Moskalenko <mom040267@gmail.com> 4 * All rights reserved. 5 * 6 * Redistribution and use in source and binary forms, with or without 7 * modification, are permitted provided that the following conditions 8 * are met: 9 * 1. Redistributions of source code must retain the above copyright 10 * notice, this list of conditions and the following disclaimer. 11 * 2. Redistributions in binary form must reproduce the above copyright 12 * notice, this list of conditions and the following disclaimer in the 13 * documentation and/or other materials provided with the distribution. 14 * 15 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 16 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 17 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 18 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 19 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 20 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 21 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 22 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 23 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 24 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 25 * SUCH DAMAGE. 26 */ 27 28 #include <sys/cdefs.h> 29 __FBSDID("$FreeBSD$"); 30 31 #include <ctype.h> 32 #include <errno.h> 33 #include <err.h> 34 #include <langinfo.h> 35 #include <math.h> 36 #include <stdlib.h> 37 #include <string.h> 38 #include <wchar.h> 39 #include <wctype.h> 40 41 #include "bwstring.h" 42 #include "sort.h" 43 44 bool byte_sort; 45 46 static wchar_t **wmonths; 47 static unsigned char **cmonths; 48 49 /* initialise months */ 50 51 void 52 initialise_months(void) 53 { 54 const nl_item item[12] = { ABMON_1, ABMON_2, ABMON_3, ABMON_4, 55 ABMON_5, ABMON_6, ABMON_7, ABMON_8, ABMON_9, ABMON_10, 56 ABMON_11, ABMON_12 }; 57 unsigned char *tmp; 58 size_t len; 59 60 if (MB_CUR_MAX == 1) { 61 if (cmonths == NULL) { 62 unsigned char *m; 63 64 cmonths = sort_malloc(sizeof(unsigned char*) * 12); 65 for (int i = 0; i < 12; i++) { 66 cmonths[i] = NULL; 67 tmp = (unsigned char *) nl_langinfo(item[i]); 68 if (debug_sort) 69 printf("month[%d]=%s\n", i, tmp); 70 if (*tmp == '\0') 71 continue; 72 m = sort_strdup(tmp); 73 len = strlen(tmp); 74 for (unsigned int j = 0; j < len; j++) 75 m[j] = toupper(m[j]); 76 cmonths[i] = m; 77 } 78 } 79 80 } else { 81 if (wmonths == NULL) { 82 wchar_t *m; 83 84 wmonths = sort_malloc(sizeof(wchar_t *) * 12); 85 for (int i = 0; i < 12; i++) { 86 wmonths[i] = NULL; 87 tmp = (unsigned char *) nl_langinfo(item[i]); 88 if (debug_sort) 89 printf("month[%d]=%s\n", i, tmp); 90 if (*tmp == '\0') 91 continue; 92 len = strlen(tmp); 93 m = sort_malloc(SIZEOF_WCHAR_STRING(len + 1)); 94 if (mbstowcs(m, (char*)tmp, len) == 95 ((size_t) - 1)) { 96 sort_free(m); 97 continue; 98 } 99 m[len] = L'\0'; 100 for (unsigned int j = 0; j < len; j++) 101 m[j] = towupper(m[j]); 102 wmonths[i] = m; 103 } 104 } 105 } 106 } 107 108 /* 109 * Compare two wide-character strings 110 */ 111 static int 112 wide_str_coll(const wchar_t *s1, const wchar_t *s2) 113 { 114 int ret = 0; 115 116 errno = 0; 117 ret = wcscoll(s1, s2); 118 if (errno == EILSEQ) { 119 errno = 0; 120 ret = wcscmp(s1, s2); 121 if (errno != 0) { 122 for (size_t i = 0; ; ++i) { 123 wchar_t c1 = s1[i]; 124 wchar_t c2 = s2[i]; 125 if (c1 == L'\0') 126 return ((c2 == L'\0') ? 0 : -1); 127 if (c2 == L'\0') 128 return (+1); 129 if (c1 == c2) 130 continue; 131 return ((int)(c1 - c2)); 132 } 133 } 134 } 135 return (ret); 136 } 137 138 /* counterparts of wcs functions */ 139 140 void 141 bwsprintf(FILE *f, struct bwstring *bws, const char *prefix, const char *suffix) 142 { 143 144 if (MB_CUR_MAX == 1) 145 fprintf(f, "%s%s%s", prefix, bws->data.cstr, suffix); 146 else 147 fprintf(f, "%s%S%s", prefix, bws->data.wstr, suffix); 148 } 149 150 const void* bwsrawdata(const struct bwstring *bws) 151 { 152 153 return (&(bws->data)); 154 } 155 156 size_t bwsrawlen(const struct bwstring *bws) 157 { 158 159 return ((MB_CUR_MAX == 1) ? bws->len : SIZEOF_WCHAR_STRING(bws->len)); 160 } 161 162 size_t 163 bws_memsize(const struct bwstring *bws) 164 { 165 166 return ((MB_CUR_MAX == 1) ? (bws->len + 2 + sizeof(struct bwstring)) : 167 (SIZEOF_WCHAR_STRING(bws->len + 1) + sizeof(struct bwstring))); 168 } 169 170 void 171 bws_setlen(struct bwstring *bws, size_t newlen) 172 { 173 174 if (bws && newlen != bws->len && newlen <= bws->len) { 175 bws->len = newlen; 176 if (MB_CUR_MAX == 1) 177 bws->data.cstr[newlen] = '\0'; 178 else 179 bws->data.wstr[newlen] = L'\0'; 180 } 181 } 182 183 /* 184 * Allocate a new binary string of specified size 185 */ 186 struct bwstring * 187 bwsalloc(size_t sz) 188 { 189 struct bwstring *ret; 190 191 if (MB_CUR_MAX == 1) 192 ret = sort_malloc(sizeof(struct bwstring) + 1 + sz); 193 else 194 ret = sort_malloc(sizeof(struct bwstring) + 195 SIZEOF_WCHAR_STRING(sz + 1)); 196 ret->len = sz; 197 198 if (MB_CUR_MAX == 1) 199 ret->data.cstr[ret->len] = '\0'; 200 else 201 ret->data.wstr[ret->len] = L'\0'; 202 203 return (ret); 204 } 205 206 /* 207 * Create a copy of binary string. 208 * New string size equals the length of the old string. 209 */ 210 struct bwstring * 211 bwsdup(const struct bwstring *s) 212 { 213 214 if (s == NULL) 215 return (NULL); 216 else { 217 struct bwstring *ret = bwsalloc(s->len); 218 219 if (MB_CUR_MAX == 1) 220 memcpy(ret->data.cstr, s->data.cstr, (s->len)); 221 else 222 memcpy(ret->data.wstr, s->data.wstr, 223 SIZEOF_WCHAR_STRING(s->len)); 224 225 return (ret); 226 } 227 } 228 229 /* 230 * Create a new binary string from a wide character buffer. 231 */ 232 struct bwstring * 233 bwssbdup(const wchar_t *str, size_t len) 234 { 235 236 if (str == NULL) 237 return ((len == 0) ? bwsalloc(0) : NULL); 238 else { 239 struct bwstring *ret; 240 241 ret = bwsalloc(len); 242 243 if (MB_CUR_MAX == 1) 244 for (size_t i = 0; i < len; ++i) 245 ret->data.cstr[i] = (unsigned char) str[i]; 246 else 247 memcpy(ret->data.wstr, str, SIZEOF_WCHAR_STRING(len)); 248 249 return (ret); 250 } 251 } 252 253 /* 254 * Create a new binary string from a raw binary buffer. 255 */ 256 struct bwstring * 257 bwscsbdup(const unsigned char *str, size_t len) 258 { 259 struct bwstring *ret; 260 261 ret = bwsalloc(len); 262 263 if (str) { 264 if (MB_CUR_MAX == 1) 265 memcpy(ret->data.cstr, str, len); 266 else { 267 mbstate_t mbs; 268 const char *s; 269 size_t charlen, chars, cptr; 270 271 charlen = chars = 0; 272 cptr = 0; 273 s = (const char *) str; 274 275 memset(&mbs, 0, sizeof(mbs)); 276 277 while (cptr < len) { 278 size_t n = MB_CUR_MAX; 279 280 if (n > len - cptr) 281 n = len - cptr; 282 charlen = mbrlen(s + cptr, n, &mbs); 283 switch (charlen) { 284 case 0: 285 /* FALLTHROUGH */ 286 case (size_t) -1: 287 /* FALLTHROUGH */ 288 case (size_t) -2: 289 ret->data.wstr[chars++] = 290 (unsigned char) s[cptr]; 291 ++cptr; 292 break; 293 default: 294 n = mbrtowc(ret->data.wstr + (chars++), 295 s + cptr, charlen, &mbs); 296 if ((n == (size_t)-1) || (n == (size_t)-2)) 297 /* NOTREACHED */ 298 err(2, "mbrtowc error"); 299 cptr += charlen; 300 }; 301 } 302 303 ret->len = chars; 304 ret->data.wstr[ret->len] = L'\0'; 305 } 306 } 307 return (ret); 308 } 309 310 /* 311 * De-allocate object memory 312 */ 313 void 314 bwsfree(const struct bwstring *s) 315 { 316 317 if (s) 318 sort_free(s); 319 } 320 321 /* 322 * Copy content of src binary string to dst. 323 * If the capacity of the dst string is not sufficient, 324 * then the data is truncated. 325 */ 326 size_t 327 bwscpy(struct bwstring *dst, const struct bwstring *src) 328 { 329 size_t nums = src->len; 330 331 if (nums > dst->len) 332 nums = dst->len; 333 dst->len = nums; 334 335 if (MB_CUR_MAX == 1) { 336 memcpy(dst->data.cstr, src->data.cstr, nums); 337 dst->data.cstr[dst->len] = '\0'; 338 } else { 339 memcpy(dst->data.wstr, src->data.wstr, 340 SIZEOF_WCHAR_STRING(nums + 1)); 341 dst->data.wstr[dst->len] = L'\0'; 342 } 343 344 return (nums); 345 } 346 347 /* 348 * Copy content of src binary string to dst, 349 * with specified number of symbols to be copied. 350 * If the capacity of the dst string is not sufficient, 351 * then the data is truncated. 352 */ 353 struct bwstring * 354 bwsncpy(struct bwstring *dst, const struct bwstring *src, size_t size) 355 { 356 size_t nums = src->len; 357 358 if (nums > dst->len) 359 nums = dst->len; 360 if (nums > size) 361 nums = size; 362 dst->len = nums; 363 364 if (MB_CUR_MAX == 1) { 365 memcpy(dst->data.cstr, src->data.cstr, nums); 366 dst->data.cstr[dst->len] = '\0'; 367 } else { 368 memcpy(dst->data.wstr, src->data.wstr, 369 SIZEOF_WCHAR_STRING(nums + 1)); 370 dst->data.wstr[dst->len] = L'\0'; 371 } 372 373 return (dst); 374 } 375 376 /* 377 * Copy content of src binary string to dst, 378 * with specified number of symbols to be copied. 379 * An offset value can be specified, from the start of src string. 380 * If the capacity of the dst string is not sufficient, 381 * then the data is truncated. 382 */ 383 struct bwstring * 384 bwsnocpy(struct bwstring *dst, const struct bwstring *src, size_t offset, 385 size_t size) 386 { 387 388 if (offset >= src->len) { 389 dst->data.wstr[0] = 0; 390 dst->len = 0; 391 } else { 392 size_t nums = src->len - offset; 393 394 if (nums > dst->len) 395 nums = dst->len; 396 if (nums > size) 397 nums = size; 398 dst->len = nums; 399 if (MB_CUR_MAX == 1) { 400 memcpy(dst->data.cstr, src->data.cstr + offset, 401 (nums)); 402 dst->data.cstr[dst->len] = '\0'; 403 } else { 404 memcpy(dst->data.wstr, src->data.wstr + offset, 405 SIZEOF_WCHAR_STRING(nums)); 406 dst->data.wstr[dst->len] = L'\0'; 407 } 408 } 409 return (dst); 410 } 411 412 /* 413 * Write binary string to the file. 414 * The output is ended either with '\n' (nl == true) 415 * or '\0' (nl == false). 416 */ 417 size_t 418 bwsfwrite(struct bwstring *bws, FILE *f, bool zero_ended) 419 { 420 421 if (MB_CUR_MAX == 1) { 422 size_t len = bws->len; 423 424 if (!zero_ended) { 425 bws->data.cstr[len] = '\n'; 426 427 if (fwrite(bws->data.cstr, len + 1, 1, f) < 1) 428 err(2, NULL); 429 430 bws->data.cstr[len] = '\0'; 431 } else if (fwrite(bws->data.cstr, len + 1, 1, f) < 1) 432 err(2, NULL); 433 434 return (len + 1); 435 436 } else { 437 wchar_t eols; 438 size_t printed = 0; 439 440 eols = zero_ended ? btowc('\0') : btowc('\n'); 441 442 while (printed < BWSLEN(bws)) { 443 const wchar_t *s = bws->data.wstr + printed; 444 445 if (*s == L'\0') { 446 int nums; 447 448 nums = fwprintf(f, L"%lc", *s); 449 450 if (nums != 1) 451 err(2, NULL); 452 ++printed; 453 } else { 454 int nums; 455 456 nums = fwprintf(f, L"%ls", s); 457 458 if (nums < 1) 459 err(2, NULL); 460 printed += nums; 461 } 462 } 463 fwprintf(f, L"%lc", eols); 464 return (printed + 1); 465 } 466 } 467 468 /* 469 * Allocate and read a binary string from file. 470 * The strings are nl-ended or zero-ended, depending on the sort setting. 471 */ 472 struct bwstring * 473 bwsfgetln(FILE *f, size_t *len, bool zero_ended, struct reader_buffer *rb) 474 { 475 wint_t eols; 476 477 eols = zero_ended ? btowc('\0') : btowc('\n'); 478 479 if (!zero_ended && (MB_CUR_MAX > 1)) { 480 wchar_t *ret; 481 482 ret = fgetwln(f, len); 483 484 if (ret == NULL) { 485 if (!feof(f)) 486 err(2, NULL); 487 return (NULL); 488 } 489 if (*len > 0) { 490 if (ret[*len - 1] == (wchar_t)eols) 491 --(*len); 492 } 493 return (bwssbdup(ret, *len)); 494 495 } else if (!zero_ended && (MB_CUR_MAX == 1)) { 496 char *ret; 497 498 ret = fgetln(f, len); 499 500 if (ret == NULL) { 501 if (!feof(f)) 502 err(2, NULL); 503 return (NULL); 504 } 505 if (*len > 0) { 506 if (ret[*len - 1] == '\n') 507 --(*len); 508 } 509 return (bwscsbdup((unsigned char*)ret, *len)); 510 511 } else { 512 *len = 0; 513 514 if (feof(f)) 515 return (NULL); 516 517 if (2 >= rb->fgetwln_z_buffer_size) { 518 rb->fgetwln_z_buffer_size += 256; 519 rb->fgetwln_z_buffer = sort_realloc(rb->fgetwln_z_buffer, 520 sizeof(wchar_t) * rb->fgetwln_z_buffer_size); 521 } 522 rb->fgetwln_z_buffer[*len] = 0; 523 524 if (MB_CUR_MAX == 1) 525 while (!feof(f)) { 526 int c; 527 528 c = fgetc(f); 529 530 if (c == EOF) { 531 if (*len == 0) 532 return (NULL); 533 goto line_read_done; 534 } 535 if (c == eols) 536 goto line_read_done; 537 538 if (*len + 1 >= rb->fgetwln_z_buffer_size) { 539 rb->fgetwln_z_buffer_size += 256; 540 rb->fgetwln_z_buffer = sort_realloc(rb->fgetwln_z_buffer, 541 SIZEOF_WCHAR_STRING(rb->fgetwln_z_buffer_size)); 542 } 543 544 rb->fgetwln_z_buffer[*len] = c; 545 rb->fgetwln_z_buffer[++(*len)] = 0; 546 } 547 else 548 while (!feof(f)) { 549 wint_t c = 0; 550 551 c = fgetwc(f); 552 553 if (c == WEOF) { 554 if (*len == 0) 555 return (NULL); 556 goto line_read_done; 557 } 558 if (c == eols) 559 goto line_read_done; 560 561 if (*len + 1 >= rb->fgetwln_z_buffer_size) { 562 rb->fgetwln_z_buffer_size += 256; 563 rb->fgetwln_z_buffer = sort_realloc(rb->fgetwln_z_buffer, 564 SIZEOF_WCHAR_STRING(rb->fgetwln_z_buffer_size)); 565 } 566 567 rb->fgetwln_z_buffer[*len] = c; 568 rb->fgetwln_z_buffer[++(*len)] = 0; 569 } 570 571 line_read_done: 572 /* we do not count the last 0 */ 573 return (bwssbdup(rb->fgetwln_z_buffer, *len)); 574 } 575 } 576 577 int 578 bwsncmp(const struct bwstring *bws1, const struct bwstring *bws2, 579 size_t offset, size_t len) 580 { 581 size_t cmp_len, len1, len2; 582 int res = 0; 583 584 cmp_len = 0; 585 len1 = bws1->len; 586 len2 = bws2->len; 587 588 if (len1 <= offset) { 589 return ((len2 <= offset) ? 0 : -1); 590 } else { 591 if (len2 <= offset) 592 return (+1); 593 else { 594 len1 -= offset; 595 len2 -= offset; 596 597 cmp_len = len1; 598 599 if (len2 < cmp_len) 600 cmp_len = len2; 601 602 if (len < cmp_len) 603 cmp_len = len; 604 605 if (MB_CUR_MAX == 1) { 606 const unsigned char *s1, *s2; 607 608 s1 = bws1->data.cstr + offset; 609 s2 = bws2->data.cstr + offset; 610 611 res = memcmp(s1, s2, cmp_len); 612 613 } else { 614 const wchar_t *s1, *s2; 615 616 s1 = bws1->data.wstr + offset; 617 s2 = bws2->data.wstr + offset; 618 619 res = memcmp(s1, s2, SIZEOF_WCHAR_STRING(cmp_len)); 620 } 621 } 622 } 623 624 if (res == 0) { 625 if (len1 < cmp_len && len1 < len2) 626 res = -1; 627 else if (len2 < cmp_len && len2 < len1) 628 res = +1; 629 } 630 631 return (res); 632 } 633 634 int 635 bwscmp(const struct bwstring *bws1, const struct bwstring *bws2, size_t offset) 636 { 637 size_t len1, len2, cmp_len; 638 int res; 639 640 len1 = bws1->len; 641 len2 = bws2->len; 642 643 len1 -= offset; 644 len2 -= offset; 645 646 cmp_len = len1; 647 648 if (len2 < cmp_len) 649 cmp_len = len2; 650 651 res = bwsncmp(bws1, bws2, offset, cmp_len); 652 653 if (res == 0) { 654 if( len1 < len2) 655 res = -1; 656 else if (len2 < len1) 657 res = +1; 658 } 659 660 return (res); 661 } 662 663 int 664 bws_iterator_cmp(bwstring_iterator iter1, bwstring_iterator iter2, size_t len) 665 { 666 wchar_t c1, c2; 667 size_t i = 0; 668 669 for (i = 0; i < len; ++i) { 670 c1 = bws_get_iter_value(iter1); 671 c2 = bws_get_iter_value(iter2); 672 if (c1 != c2) 673 return (c1 - c2); 674 iter1 = bws_iterator_inc(iter1, 1); 675 iter2 = bws_iterator_inc(iter2, 1); 676 } 677 678 return (0); 679 } 680 681 int 682 bwscoll(const struct bwstring *bws1, const struct bwstring *bws2, size_t offset) 683 { 684 size_t len1, len2; 685 686 len1 = bws1->len; 687 len2 = bws2->len; 688 689 if (len1 <= offset) 690 return ((len2 <= offset) ? 0 : -1); 691 else { 692 if (len2 <= offset) 693 return (+1); 694 else { 695 len1 -= offset; 696 len2 -= offset; 697 698 if (MB_CUR_MAX == 1) { 699 const unsigned char *s1, *s2; 700 701 s1 = bws1->data.cstr + offset; 702 s2 = bws2->data.cstr + offset; 703 704 if (byte_sort) { 705 int res = 0; 706 707 if (len1 > len2) { 708 res = memcmp(s1, s2, len2); 709 if (!res) 710 res = +1; 711 } else if (len1 < len2) { 712 res = memcmp(s1, s2, len1); 713 if (!res) 714 res = -1; 715 } else 716 res = memcmp(s1, s2, len1); 717 718 return (res); 719 720 } else { 721 int res = 0; 722 size_t i, maxlen; 723 724 i = 0; 725 maxlen = len1; 726 727 if (maxlen > len2) 728 maxlen = len2; 729 730 while (i < maxlen) { 731 /* goto next non-zero part: */ 732 while ((i < maxlen) && 733 !s1[i] && !s2[i]) 734 ++i; 735 736 if (i >= maxlen) 737 break; 738 739 if (s1[i] == 0) { 740 if (s2[i] == 0) 741 /* NOTREACHED */ 742 err(2, "bwscoll error 01"); 743 else 744 return (-1); 745 } else if (s2[i] == 0) 746 return (+1); 747 748 res = strcoll((const char*)(s1 + i), (const char*)(s2 + i)); 749 if (res) 750 return (res); 751 752 while ((i < maxlen) && 753 s1[i] && s2[i]) 754 ++i; 755 756 if (i >= maxlen) 757 break; 758 759 if (s1[i] == 0) { 760 if (s2[i] == 0) { 761 ++i; 762 continue; 763 } else 764 return (-1); 765 } else if (s2[i] == 0) 766 return (+1); 767 else 768 /* NOTREACHED */ 769 err(2, "bwscoll error 02"); 770 } 771 772 if (len1 < len2) 773 return (-1); 774 else if (len1 > len2) 775 return (+1); 776 777 return (0); 778 } 779 } else { 780 const wchar_t *s1, *s2; 781 size_t i, maxlen; 782 int res = 0; 783 784 s1 = bws1->data.wstr + offset; 785 s2 = bws2->data.wstr + offset; 786 787 i = 0; 788 maxlen = len1; 789 790 if (maxlen > len2) 791 maxlen = len2; 792 793 while (i < maxlen) { 794 795 /* goto next non-zero part: */ 796 while ((i < maxlen) && 797 !s1[i] && !s2[i]) 798 ++i; 799 800 if (i >= maxlen) 801 break; 802 803 if (s1[i] == 0) { 804 if (s2[i] == 0) 805 /* NOTREACHED */ 806 err(2, "bwscoll error 1"); 807 else 808 return (-1); 809 } else if (s2[i] == 0) 810 return (+1); 811 812 res = wide_str_coll(s1 + i, s2 + i); 813 if (res) 814 return (res); 815 816 while ((i < maxlen) && s1[i] && s2[i]) 817 ++i; 818 819 if (i >= maxlen) 820 break; 821 822 if (s1[i] == 0) { 823 if (s2[i] == 0) { 824 ++i; 825 continue; 826 } else 827 return (-1); 828 } else if (s2[i] == 0) 829 return (+1); 830 else 831 /* NOTREACHED */ 832 err(2, "bwscoll error 2"); 833 } 834 835 if (len1 < len2) 836 return (-1); 837 else if (len1 > len2) 838 return (+1); 839 840 return (0); 841 } 842 } 843 } 844 } 845 846 /* 847 * Correction of the system API 848 */ 849 double 850 bwstod(struct bwstring *s0, bool *empty) 851 { 852 double ret = 0; 853 854 if (MB_CUR_MAX == 1) { 855 unsigned char *end, *s; 856 char *ep; 857 858 s = s0->data.cstr; 859 end = s + s0->len; 860 ep = NULL; 861 862 while (isblank(*s) && s < end) 863 ++s; 864 865 if (!isprint(*s)) { 866 *empty = true; 867 return (0); 868 } 869 870 ret = strtod((char*)s, &ep); 871 if ((unsigned char*) ep == s) { 872 *empty = true; 873 return (0); 874 } 875 } else { 876 wchar_t *end, *ep, *s; 877 878 s = s0->data.wstr; 879 end = s + s0->len; 880 ep = NULL; 881 882 while (iswblank(*s) && s < end) 883 ++s; 884 885 if (!iswprint(*s)) { 886 *empty = true; 887 return (0); 888 } 889 890 ret = wcstod(s, &ep); 891 if (ep == s) { 892 *empty = true; 893 return (0); 894 } 895 } 896 897 *empty = false; 898 return (ret); 899 } 900 901 /* 902 * A helper function for monthcoll. If a line matches 903 * a month name, it returns (number of the month - 1), 904 * while if there is no match, it just return -1. 905 */ 906 907 int 908 bws_month_score(const struct bwstring *s0) 909 { 910 911 if (MB_CUR_MAX == 1) { 912 const unsigned char *end, *s; 913 size_t len; 914 915 s = s0->data.cstr; 916 end = s + s0->len; 917 918 while (isblank(*s) && s < end) 919 ++s; 920 921 len = strlen((const char*)s); 922 923 for (int i = 11; i >= 0; --i) { 924 if (cmonths[i] && 925 (s == (unsigned char*)strstr((const char*)s, (char*)(cmonths[i])))) 926 return (i); 927 } 928 929 } else { 930 const wchar_t *end, *s; 931 size_t len; 932 933 s = s0->data.wstr; 934 end = s + s0->len; 935 936 while (iswblank(*s) && s < end) 937 ++s; 938 939 len = wcslen(s); 940 941 for (int i = 11; i >= 0; --i) { 942 if (wmonths[i] && (s == wcsstr(s, wmonths[i]))) 943 return (i); 944 } 945 } 946 947 return (-1); 948 } 949 950 /* 951 * Rips out leading blanks (-b). 952 */ 953 struct bwstring * 954 ignore_leading_blanks(struct bwstring *str) 955 { 956 957 if (MB_CUR_MAX == 1) { 958 unsigned char *dst, *end, *src; 959 960 src = str->data.cstr; 961 dst = src; 962 end = src + str->len; 963 964 while (src < end && isblank(*src)) 965 ++src; 966 967 if (src != dst) { 968 size_t newlen; 969 970 newlen = BWSLEN(str) - (src - dst); 971 972 while (src < end) { 973 *dst = *src; 974 ++dst; 975 ++src; 976 } 977 bws_setlen(str, newlen); 978 } 979 } else { 980 wchar_t *dst, *end, *src; 981 982 src = str->data.wstr; 983 dst = src; 984 end = src + str->len; 985 986 while (src < end && iswblank(*src)) 987 ++src; 988 989 if (src != dst) { 990 991 size_t newlen = BWSLEN(str) - (src - dst); 992 993 while (src < end) { 994 *dst = *src; 995 ++dst; 996 ++src; 997 } 998 bws_setlen(str, newlen); 999 1000 } 1001 } 1002 return (str); 1003 } 1004 1005 /* 1006 * Rips out nonprinting characters (-i). 1007 */ 1008 struct bwstring * 1009 ignore_nonprinting(struct bwstring *str) 1010 { 1011 size_t newlen = str->len; 1012 1013 if (MB_CUR_MAX == 1) { 1014 unsigned char *dst, *end, *src; 1015 unsigned char c; 1016 1017 src = str->data.cstr; 1018 dst = src; 1019 end = src + str->len; 1020 1021 while (src < end) { 1022 c = *src; 1023 if (isprint(c)) { 1024 *dst = c; 1025 ++dst; 1026 ++src; 1027 } else { 1028 ++src; 1029 --newlen; 1030 } 1031 } 1032 } else { 1033 wchar_t *dst, *end, *src; 1034 wchar_t c; 1035 1036 src = str->data.wstr; 1037 dst = src; 1038 end = src + str->len; 1039 1040 while (src < end) { 1041 c = *src; 1042 if (iswprint(c)) { 1043 *dst = c; 1044 ++dst; 1045 ++src; 1046 } else { 1047 ++src; 1048 --newlen; 1049 } 1050 } 1051 } 1052 bws_setlen(str, newlen); 1053 1054 return (str); 1055 } 1056 1057 /* 1058 * Rips out any characters that are not alphanumeric characters 1059 * nor blanks (-d). 1060 */ 1061 struct bwstring * 1062 dictionary_order(struct bwstring *str) 1063 { 1064 size_t newlen = str->len; 1065 1066 if (MB_CUR_MAX == 1) { 1067 unsigned char *dst, *end, *src; 1068 unsigned char c; 1069 1070 src = str->data.cstr; 1071 dst = src; 1072 end = src + str->len; 1073 1074 while (src < end) { 1075 c = *src; 1076 if (isalnum(c) || isblank(c)) { 1077 *dst = c; 1078 ++dst; 1079 ++src; 1080 } else { 1081 ++src; 1082 --newlen; 1083 } 1084 } 1085 } else { 1086 wchar_t *dst, *end, *src; 1087 wchar_t c; 1088 1089 src = str->data.wstr; 1090 dst = src; 1091 end = src + str->len; 1092 1093 while (src < end) { 1094 c = *src; 1095 if (iswalnum(c) || iswblank(c)) { 1096 *dst = c; 1097 ++dst; 1098 ++src; 1099 } else { 1100 ++src; 1101 --newlen; 1102 } 1103 } 1104 } 1105 bws_setlen(str, newlen); 1106 1107 return (str); 1108 } 1109 1110 /* 1111 * Converts string to lower case(-f). 1112 */ 1113 struct bwstring * 1114 ignore_case(struct bwstring *str) 1115 { 1116 1117 if (MB_CUR_MAX == 1) { 1118 unsigned char *end, *s; 1119 1120 s = str->data.cstr; 1121 end = s + str->len; 1122 1123 while (s < end) { 1124 *s = toupper(*s); 1125 ++s; 1126 } 1127 } else { 1128 wchar_t *end, *s; 1129 1130 s = str->data.wstr; 1131 end = s + str->len; 1132 1133 while (s < end) { 1134 *s = towupper(*s); 1135 ++s; 1136 } 1137 } 1138 return (str); 1139 } 1140 1141 void 1142 bws_disorder_warnx(struct bwstring *s, const char *fn, size_t pos) 1143 { 1144 1145 if (MB_CUR_MAX == 1) 1146 warnx("%s:%zu: disorder: %s", fn, pos + 1, s->data.cstr); 1147 else 1148 warnx("%s:%zu: disorder: %ls", fn, pos + 1, s->data.wstr); 1149 } 1150