1 /* 2 * This file and its contents are supplied under the terms of the 3 * Common Development and Distribution License ("CDDL"), version 1.0. 4 * You may only use this file in accordance with the terms of version 5 * 1.0 of the CDDL. 6 * 7 * A full copy of the text of the CDDL should have accompanied this 8 * source. A copy of the CDDL is also available via the Internet at 9 * http://www.illumos.org/license/CDDL. 10 */ 11 12 /* 13 * Copyright 2020 Robert Mustacchi 14 */ 15 16 /* 17 * Test the implementation of various pieces of uchar.h(3HEAD) functionality. 18 */ 19 20 #include <locale.h> 21 #include <err.h> 22 #include <stdlib.h> 23 #include <sys/types.h> 24 #include <sys/sysmacros.h> 25 #include <strings.h> 26 #include <wchar.h> 27 #include <uchar.h> 28 #include <errno.h> 29 30 static const char *uchar_wide = "光"; 31 static const char32_t uchar_value = 0x5149; 32 static const char *uchar_hello = "hello"; 33 34 static void 35 update_locale(const char *loc) 36 { 37 const char *newloc = setlocale(LC_CTYPE, loc); 38 if (newloc == NULL) { 39 err(EXIT_FAILURE, "TEST FAILED: failed to update locale to %s", 40 loc); 41 } 42 43 if (strcmp(newloc, loc) != 0) { 44 errx(EXIT_FAILURE, "TEST FAILED: locale set to %s, but got %s", 45 loc, newloc); 46 } 47 } 48 49 static boolean_t 50 mbrtoc32_ascii(mbstate_t *mbs) 51 { 52 char32_t out; 53 size_t len; 54 boolean_t ret = B_TRUE; 55 56 if ((len = mbrtoc32(&out, uchar_hello, 5, mbs)) != 1) { 57 warnx("expected mbrtoc32 to return 1, returned %zu", len); 58 ret = B_FALSE; 59 } 60 61 if (out != 'h') { 62 warnx("got bad char32_t, expected 0x%x, found 0x%x\n", 'h', 63 out); 64 ret = B_FALSE; 65 } 66 67 if ((len = mbrtoc32(&out, uchar_hello + 1, 4, mbs)) != 1) { 68 warnx("expected mbrtoc32 to return 1, returned %zu", len); 69 ret = B_FALSE; 70 } 71 72 if (out != 'e') { 73 warnx("got bad char32_t, expected 0x%x, found 0x%x\n", 'h', 74 out); 75 ret = B_FALSE; 76 } 77 78 return (ret); 79 } 80 81 static boolean_t 82 mbrtoc32_ascii_internal(void) 83 { 84 return (mbrtoc32_ascii(NULL)); 85 } 86 87 static boolean_t 88 mbrtoc32_ascii_mbstate(void) 89 { 90 mbstate_t mbs; 91 92 bzero(&mbs, sizeof (mbs)); 93 return (mbrtoc32_ascii(&mbs)); 94 } 95 96 static boolean_t 97 mbrtoc32_badseq_utf8(void) 98 { 99 mbstate_t mbs; 100 size_t len; 101 char32_t out; 102 boolean_t ret = B_TRUE; 103 char *badstr; 104 105 bzero(&mbs, sizeof (mbs)); 106 len = mbrtoc32(&out, "\xa9", 1, &mbs); 107 if (len != (size_t)-1) { 108 warnx("mbrtoc32 returned %zu, not %zu", len, (size_t)-1); 109 ret = B_FALSE; 110 } 111 112 if (errno != EILSEQ) { 113 warnx("found bad errno, expected %d, found %d\n", errno, 114 EILSEQ); 115 ret = B_FALSE; 116 } 117 118 badstr = strdup(uchar_wide); 119 if (badstr == NULL) { 120 warn("failed to duplicate uchar_wide"); 121 return (B_FALSE); 122 } 123 124 badstr[1] = '?'; 125 bzero(&mbs, sizeof (mbs)); 126 len = mbrtoc32(&out, badstr, strlen(badstr), &mbs); 127 free(badstr); 128 if (len != (size_t)-1) { 129 warnx("mbrtoc32 returned %zu, not %zu", len, (size_t)-1); 130 ret = B_FALSE; 131 } 132 133 if (errno != EILSEQ) { 134 warnx("found bad errno, expected %d, found %d\n", errno, 135 EILSEQ); 136 ret = B_FALSE; 137 } 138 139 return (ret); 140 } 141 142 static boolean_t 143 mbrtoc32_roundtrip(void) 144 { 145 char32_t out; 146 size_t len, clen; 147 mbstate_t mbs; 148 char buf[MB_CUR_MAX]; 149 boolean_t ret = B_TRUE; 150 151 bzero(&mbs, sizeof (mbs)); 152 len = mbrtoc32(&out, uchar_wide, strlen(uchar_wide), &mbs); 153 if (len != 3) { 154 warnx("mbrtoc32 returned %zu, expected %u", len, 3); 155 ret = B_FALSE; 156 } 157 158 if (out != uchar_value) { 159 warnx("mbrtoc32 converted character to 0x%x not 0x%x", 160 out, uchar_value); 161 ret = B_FALSE; 162 } 163 164 clen = c32rtomb(buf, out, &mbs); 165 if (clen != len) { 166 warnx("c32rtomb returned %zu bytes, but we originally used %zu", 167 clen, len); 168 ret = B_FALSE; 169 } 170 171 if (strncmp(buf, uchar_wide, len) != 0) { 172 warnx("round trip string comparison failed"); 173 ret = B_FALSE; 174 } 175 176 return (ret); 177 } 178 179 static boolean_t 180 mbrtoc32_partial(void) 181 { 182 char32_t out; 183 size_t len, i; 184 mbstate_t mbs; 185 boolean_t ret = B_TRUE; 186 187 bzero(&mbs, sizeof (mbs)); 188 for (i = 0; i < strlen(uchar_wide) - 1; i++) { 189 len = mbrtoc32(&out, uchar_wide + i, 1, &mbs); 190 if (len != (size_t)-2) { 191 warnx("partial mbrtoc32 returned %zu, not -2", len); 192 ret = B_FALSE; 193 } 194 } 195 196 len = mbrtoc32(&out, uchar_wide + i, 1, &mbs); 197 if (len != 1) { 198 warnx("partial mbrtoc32 returned %zu, not 1", len); 199 ret = B_FALSE; 200 } 201 202 if (out != uchar_value) { 203 warnx("mbrtoc32 converted character to 0x%x not 0x%x", 204 out, uchar_value); 205 ret = B_FALSE; 206 } 207 208 return (ret); 209 } 210 211 static boolean_t 212 mbrtoc32_zero(void) 213 { 214 char32_t out, exp = L'\0'; 215 size_t len; 216 mbstate_t mbs; 217 boolean_t ret = B_TRUE; 218 219 bzero(&mbs, sizeof (mbs)); 220 len = mbrtoc32(&out, "", 1, &mbs); 221 if (len != 0) { 222 warnx("partial mbrtoc32 returned %zu, not 0", len); 223 ret = B_FALSE; 224 } 225 226 if (out != exp) { 227 warnx("mbrtoc32 converted character to 0x%x not 0x%x", 228 out, exp); 229 ret = B_FALSE; 230 } 231 232 return (ret); 233 } 234 235 static boolean_t 236 mbrtoc32_zero_len(void) 237 { 238 char32_t out = 0x12345, exp = 0x12345; 239 size_t len; 240 mbstate_t mbs; 241 boolean_t ret = B_TRUE; 242 243 bzero(&mbs, sizeof (mbs)); 244 len = mbrtoc32(&out, uchar_wide, 0, &mbs); 245 if (len != (size_t)-2) { 246 warnx("partial mbrtoc32 returned %zu, not -2", len); 247 ret = B_FALSE; 248 } 249 250 if (out != exp) { 251 warnx("mbrtoc32 incorrectly wrote to char32_t value with " 252 "zero string, found 0x%x not 0x%x", out, exp); 253 ret = B_FALSE; 254 } 255 256 return (ret); 257 } 258 259 static boolean_t 260 mbrtoc32_null(void) 261 { 262 char32_t out = 0x123456, exp = 0x123456; 263 size_t len; 264 mbstate_t mbs; 265 boolean_t ret = B_TRUE; 266 267 bzero(&mbs, sizeof (mbs)); 268 len = mbrtoc32(&out, NULL, 1, &mbs); 269 if (len != 0) { 270 warnx("partial mbrtoc32 returned %zu, not 0", len); 271 ret = B_FALSE; 272 } 273 274 if (out != exp) { 275 warnx("mbrtoc32 incorrectly wrote to char32_t value with " 276 "null string, found 0x%x not 0x%x", out, exp); 277 ret = B_FALSE; 278 } 279 280 return (ret); 281 } 282 283 static boolean_t 284 mbrtoc16_ascii(mbstate_t *mbs) 285 { 286 char16_t out; 287 size_t len; 288 boolean_t ret = B_TRUE; 289 290 if ((len = mbrtoc16(&out, uchar_hello, 5, mbs)) != 1) { 291 warnx("expected mbrtoc16 to return 1, returned %zu", len); 292 ret = B_FALSE; 293 } 294 295 if (out != 'h') { 296 warnx("got bad char16_t, expected 0x%x, found 0x%x\n", 'h', 297 out); 298 ret = B_FALSE; 299 } 300 301 if ((len = mbrtoc16(&out, uchar_hello + 1, 4, mbs)) != 1) { 302 warnx("expected mbrtoc16 to return 1, returned %zu", len); 303 ret = B_FALSE; 304 } 305 306 if (out != 'e') { 307 warnx("got bad char16_t, expected 0x%x, found 0x%x\n", 'h', 308 out); 309 ret = B_FALSE; 310 } 311 312 return (ret); 313 } 314 315 static boolean_t 316 mbrtoc16_ascii_internal(void) 317 { 318 return (mbrtoc16_ascii(NULL)); 319 } 320 321 static boolean_t 322 mbrtoc16_ascii_mbstate(void) 323 { 324 mbstate_t mbs; 325 326 bzero(&mbs, sizeof (mbs)); 327 return (mbrtoc16_ascii(&mbs)); 328 } 329 330 static boolean_t 331 mbrtoc16_null(void) 332 { 333 char16_t out = 0x1234, exp = 0x1234; 334 size_t len; 335 mbstate_t mbs; 336 boolean_t ret = B_TRUE; 337 338 bzero(&mbs, sizeof (mbs)); 339 len = mbrtoc16(&out, NULL, 1, &mbs); 340 if (len != 0) { 341 warnx("partial mbrtoc16 returned %zu, not 0", len); 342 ret = B_FALSE; 343 } 344 345 if (out != exp) { 346 warnx("mbrtoc16 incorrectly wrote to char16_t value with " 347 "null string, found 0x%x not 0x%x", out, exp); 348 ret = B_FALSE; 349 } 350 351 return (ret); 352 } 353 354 static boolean_t 355 mbrtoc16_zero(void) 356 { 357 char16_t out, exp = L'\0'; 358 size_t len; 359 mbstate_t mbs; 360 boolean_t ret = B_TRUE; 361 362 bzero(&mbs, sizeof (mbs)); 363 len = mbrtoc16(&out, "", 1, &mbs); 364 if (len != 0) { 365 warnx("partial mbrtoc16 returned %zu, not 0", len); 366 ret = B_FALSE; 367 } 368 369 if (out != exp) { 370 warnx("mbrtoc16 converted character to 0x%x not 0x%x", 371 out, exp); 372 ret = B_FALSE; 373 } 374 375 return (ret); 376 } 377 378 static boolean_t 379 mbrtoc16_zero_len(void) 380 { 381 char16_t out = 0x5432, exp = 0x5432; 382 size_t len; 383 mbstate_t mbs; 384 boolean_t ret = B_TRUE; 385 386 bzero(&mbs, sizeof (mbs)); 387 len = mbrtoc16(&out, uchar_wide, 0, &mbs); 388 if (len != (size_t)-2) { 389 warnx("partial mbrtoc16 returned %zu, not -2", len); 390 ret = B_FALSE; 391 } 392 393 if (out != exp) { 394 warnx("mbrtoc16 incorrectly wrote to char16_t value with " 395 "zero length string, found 0x%x not 0x%x", out, exp); 396 ret = B_FALSE; 397 } 398 399 return (ret); 400 } 401 402 static boolean_t 403 mbrtoc16_roundtrip(void) 404 { 405 char16_t out; 406 size_t len, clen; 407 mbstate_t mbs; 408 char buf[MB_CUR_MAX]; 409 boolean_t ret = B_TRUE; 410 411 bzero(&mbs, sizeof (mbs)); 412 len = mbrtoc16(&out, uchar_wide, strlen(uchar_wide), &mbs); 413 if (len != 3) { 414 warnx("mbrtoc16 returned %zu, expected %u", len, 3); 415 ret = B_FALSE; 416 } 417 418 if (out != uchar_value) { 419 warnx("mbrtoc16 converted character to 0x%x not 0x%x", 420 out, uchar_value); 421 ret = B_FALSE; 422 } 423 424 clen = c16rtomb(buf, out, &mbs); 425 if (clen != len) { 426 warnx("c16rtomb returned %zu bytes, but we originally used %zu", 427 clen, len); 428 ret = B_FALSE; 429 } 430 431 if (strncmp(buf, uchar_wide, len) != 0) { 432 warnx("round trip string comparison failed"); 433 ret = B_FALSE; 434 } 435 436 return (ret); 437 } 438 439 static boolean_t 440 mbrtoc16_partial(void) 441 { 442 char16_t out; 443 size_t len, i; 444 mbstate_t mbs; 445 boolean_t ret = B_TRUE; 446 447 bzero(&mbs, sizeof (mbs)); 448 for (i = 0; i < strlen(uchar_wide) - 1; i++) { 449 len = mbrtoc16(&out, uchar_wide + i, 1, &mbs); 450 if (len != (size_t)-2) { 451 warnx("partial mbrtoc16 returned %zu, not -2", len); 452 ret = B_FALSE; 453 } 454 } 455 456 len = mbrtoc16(&out, uchar_wide + i, 1, &mbs); 457 if (len != 1) { 458 warnx("partial mbrtoc16 returned %zu, not 1", len); 459 ret = B_FALSE; 460 } 461 462 if (out != uchar_value) { 463 warnx("mbrtoc16 converted character to 0x%x not 0x%x", 464 out, uchar_value); 465 ret = B_FALSE; 466 } 467 468 return (ret); 469 } 470 471 static boolean_t 472 mbrtoc16_surrogate(void) 473 { 474 char16_t out0, out1; 475 size_t len, clen; 476 mbstate_t mbs; 477 const char *surrogate = "\xF0\x9F\x92\xA9"; 478 char16_t exp0 = 0xd83d, exp1 = 0xdca9; 479 size_t slen = strlen(surrogate); 480 boolean_t ret = B_TRUE; 481 char buf[MB_CUR_MAX]; 482 483 bzero(&mbs, sizeof (mbs)); 484 len = mbrtoc16(&out0, surrogate, slen, &mbs); 485 if (len != slen) { 486 warnx("mbrtoc16 returned %zu, expected %zu", len, slen); 487 ret = B_FALSE; 488 } 489 490 if (out0 != exp0) { 491 warnx("mbrtoc16 converted character to 0x%x not 0x%x", 492 out0, exp0); 493 ret = B_FALSE; 494 } 495 496 if (mbsinit(&mbs) != 0) { 497 warnx("mb state with a surrogate character is somehow in the " 498 "initial state"); 499 ret = B_FALSE; 500 } 501 502 len = mbrtoc16(&out1, uchar_wide, strlen(uchar_wide), &mbs); 503 if (len != (size_t)-3) { 504 warnx("mbrtoc16 returned %zu, expected -3", len); 505 ret = B_FALSE; 506 } 507 508 if (mbsinit(&mbs) == 0) { 509 warnx("mb state with after both surrogate characters isn't " 510 "in initial state"); 511 ret = B_FALSE; 512 } 513 514 if (out1 != exp1) { 515 warnx("mbrtoc32 converted character to 0x%x not 0x%x", 516 out1, exp1); 517 ret = B_FALSE; 518 } 519 520 clen = c16rtomb(buf, out0, &mbs); 521 if (clen != 0) { 522 warnx("c16rtomb returned %zu bytes, but expected zero for the " 523 "first surrogate", clen); 524 ret = B_FALSE; 525 } 526 527 if (mbsinit(&mbs) != 0) { 528 warnx("mb state with a surrogate character is somehow in the " 529 "initial state"); 530 ret = B_FALSE; 531 } 532 533 clen = c16rtomb(buf, out1, &mbs); 534 if (clen != slen) { 535 warnx("c16rtomb returned %zu, expected %zu", len, slen); 536 ret = B_FALSE; 537 } 538 539 if (mbsinit(&mbs) == 0) { 540 warnx("mb state with after both surrogate characters isn't " 541 "in initial state"); 542 ret = B_FALSE; 543 } 544 545 if (strncmp(buf, surrogate, slen) != 0) { 546 warnx("round trip string comparison failed"); 547 ret = B_FALSE; 548 } 549 550 return (ret); 551 } 552 553 static boolean_t 554 c32rtomb_eilseq_iso8859(void) 555 { 556 char buf[MB_CUR_MAX]; 557 mbstate_t mbs; 558 size_t len; 559 boolean_t ret = B_TRUE; 560 561 bzero(&mbs, sizeof (mbs)); 562 len = c32rtomb(buf, uchar_value, &mbs); 563 if (len != (size_t)-1) { 564 warnx("c32rtomb returned %zd, expected -1\n", len); 565 ret = B_FALSE; 566 } 567 568 if (errno != EILSEQ) { 569 warnx("expected errno set to %d was %d", EILSEQ, errno); 570 ret = B_FALSE; 571 } 572 573 return (ret); 574 } 575 576 static boolean_t 577 c16rtomb_eilseq_iso8859(void) 578 { 579 char buf[MB_CUR_MAX]; 580 mbstate_t mbs; 581 size_t len; 582 boolean_t ret = B_TRUE; 583 584 bzero(&mbs, sizeof (mbs)); 585 len = c32rtomb(buf, (char16_t)uchar_value, &mbs); 586 if (len != (size_t)-1) { 587 warnx("c32rtomb returned %zd, expected -1\n", len); 588 ret = B_FALSE; 589 } 590 591 if (errno != EILSEQ) { 592 warnx("expected errno set to %d was %d", EILSEQ, errno); 593 ret = B_FALSE; 594 } 595 596 return (ret); 597 } 598 599 static boolean_t 600 c32rtomb_eilseq_utf8(void) 601 { 602 char buf[MB_CUR_MAX]; 603 mbstate_t mbs; 604 size_t len; 605 boolean_t ret = B_TRUE; 606 607 bzero(&mbs, sizeof (mbs)); 608 len = c32rtomb(buf, UINT32_MAX, &mbs); 609 if (len != (size_t)-1) { 610 warnx("c32rtomb returned %zd, expected -1\n", len); 611 ret = B_FALSE; 612 } 613 614 if (errno != EILSEQ) { 615 warnx("expected errno set to %d was %d", EILSEQ, errno); 616 ret = B_FALSE; 617 } 618 619 return (ret); 620 } 621 622 static boolean_t 623 c16rtomb_bad_first(void) 624 { 625 char buf[MB_CUR_MAX]; 626 mbstate_t mbs; 627 size_t len, i; 628 char16_t first = 0xd83d; 629 char16_t bad[] = { 0x0, 0xd7ff, 0xd83d, 0xd900, 0xffff }; 630 boolean_t ret = B_TRUE; 631 632 for (i = 0; i < ARRAY_SIZE(bad); i++) { 633 bzero(&mbs, sizeof (mbs)); 634 len = c16rtomb(buf, first, &mbs); 635 if (len != 0) { 636 warnx("c16rtomb returned %zd, expected 0\n", len); 637 ret = B_FALSE; 638 } 639 640 len = c16rtomb(buf, bad[i], &mbs); 641 if (len != (size_t)-1) { 642 warnx("c16rtomb surrogate %x returned %zd, expected " 643 "-1\n", bad[i], len); 644 ret = B_FALSE; 645 } 646 647 if (errno != EILSEQ) { 648 warnx("expected errno set to %d was %d", EILSEQ, errno); 649 ret = B_FALSE; 650 } 651 } 652 653 return (ret); 654 } 655 656 static boolean_t 657 c16rtomb_bad_second(void) 658 { 659 char buf[MB_CUR_MAX]; 660 mbstate_t mbs; 661 size_t len, i; 662 char16_t bad[] = { 0xdc00, 0xdd34, 0xdfff }; 663 boolean_t ret = B_TRUE; 664 665 for (i = 0; i < ARRAY_SIZE(bad); i++) { 666 bzero(&mbs, sizeof (mbs)); 667 len = c16rtomb(buf, bad[i], &mbs); 668 if (len != (size_t)-1) { 669 warnx("c16rtomb surrogate %x returned %zd, expected " 670 "-1\n", bad[i], len); 671 ret = B_FALSE; 672 } 673 674 if (errno != EILSEQ) { 675 warnx("expected errno set to %d was %d", EILSEQ, errno); 676 ret = B_FALSE; 677 } 678 } 679 680 return (ret); 681 } 682 683 static boolean_t 684 c32rtomb_null(void) 685 { 686 size_t len; 687 mbstate_t mbs; 688 boolean_t ret = B_TRUE; 689 690 bzero(&mbs, sizeof (mbs)); 691 len = c32rtomb(NULL, uchar_value, &mbs); 692 if (len != 1) { 693 warnx("c32rtomb returned %zu, expected %d", len, 1); 694 ret = B_FALSE; 695 } 696 697 return (ret); 698 } 699 700 static boolean_t 701 c16rtomb_null(void) 702 { 703 size_t len; 704 mbstate_t mbs; 705 boolean_t ret = B_TRUE; 706 707 bzero(&mbs, sizeof (mbs)); 708 len = c16rtomb(NULL, uchar_value, &mbs); 709 if (len != 1) { 710 warnx("c16rtomb returned %zu, expected %d", len, 1); 711 ret = B_FALSE; 712 } 713 714 return (ret); 715 } 716 717 typedef boolean_t (*uchar_test_f)(void); 718 719 typedef struct uchar_test { 720 uchar_test_f ut_func; 721 const char *ut_test; 722 const char *ut_locale; 723 } uchar_test_t; 724 725 static const uchar_test_t uchar_tests[] = { 726 { mbrtoc32_ascii_mbstate, "mbrtoc32: ascii conversion" }, 727 { mbrtoc32_ascii_internal, "mbrtoc32: ascii conversion (internal " 728 "mbstate_t)" }, 729 { mbrtoc32_badseq_utf8, "mbrtoc32: bad locale sequence (UTF-8)" }, 730 { mbrtoc32_roundtrip, "mbrtoc32: round trip conversion" }, 731 { mbrtoc32_partial, "mbrtoc32: correctly consume partial sequences" }, 732 { mbrtoc32_zero, "mbrtoc32: correctly handle L'\\0'" }, 733 { mbrtoc32_zero_len, "mbrtoc32: correctly handle length of zero" }, 734 { mbrtoc32_null, "mbrtoc32: correctly handle null string" }, 735 { mbrtoc16_ascii_mbstate, "mbrtoc16: ascii conversion" }, 736 { mbrtoc16_ascii_internal, "mbrtoc16: ascii conversion (internal " 737 "mbstate_t)" }, 738 { mbrtoc16_null, "mbrtoc16: correctly handle null string" }, 739 { mbrtoc16_zero, "mbrtoc16: correctly handle L'\\0'" }, 740 { mbrtoc16_zero_len, "mbrtoc16: correctly handle length of zero" }, 741 { mbrtoc16_roundtrip, "mbrtoc16: round trip conversion" }, 742 { mbrtoc16_partial, "mbrtoc16: correctly consume partial sequences" }, 743 { mbrtoc16_surrogate, "mbrtoc16: correctly generate surrogate pairs " 744 "and round trip conversion" }, 745 { c32rtomb_eilseq_iso8859, "c32rtomb: character outside of locale is " 746 "caught", "en_US.ISO8859-1" }, 747 { c16rtomb_eilseq_iso8859, "c16rtomb: character outside of locale is " 748 "caught", "en_US.ISO8859-1" }, 749 { c32rtomb_eilseq_utf8, "c32rtomb: character outside of locale is " 750 "caught" }, 751 { c16rtomb_bad_first, "c16rtomb: bad first surrogate pair" }, 752 { c16rtomb_bad_second, "c16rtomb: bad second surrogate pair" }, 753 { c32rtomb_null, "c32rtomb: correctly handle null buffer" }, 754 { c16rtomb_null, "c16rtomb: correctly handle null buffer" }, 755 }; 756 757 int 758 main(void) 759 { 760 uint_t i; 761 uint_t passes = 0; 762 uint_t ntests = ARRAY_SIZE(uchar_tests); 763 764 for (i = 0; i < ntests; i++) { 765 boolean_t r; 766 767 /* 768 * Default to a standard UTF-8 locale if none is requested by 769 * the test. 770 */ 771 if (uchar_tests[i].ut_locale != NULL) { 772 update_locale(uchar_tests[i].ut_locale); 773 } else { 774 update_locale("en_US.UTF-8"); 775 } 776 777 r = uchar_tests[i].ut_func(); 778 (void) fprintf(stderr, "TEST %s: %s\n", r ? "PASSED" : "FAILED", 779 uchar_tests[i].ut_test); 780 if (r) { 781 passes++; 782 } 783 } 784 785 (void) printf("%d/%d test%s passed\n", passes, ntests, 786 passes > 1 ? "s" : ""); 787 return (passes == ntests ? EXIT_SUCCESS : EXIT_FAILURE); 788 789 } 790