Home | History | Annotate | Download | only in tests
      1 /*
      2  * Copyright 2011 Google Inc.
      3  *
      4  * Use of this source code is governed by a BSD-style license that can be
      5  * found in the LICENSE file.
      6  */
      7 
      8 #include "Test.h"
      9 
     10 #include "SkString.h"
     11 #include "SkStringUtils.h"
     12 
     13 #include <stdarg.h>
     14 #include <stdio.h>
     15 #include <thread>
     16 
     17 static const char* gThirtyWideDecimal = "%30d";
     18 
     19 DEF_TEST(String, reporter) {
     20     SkString    a;
     21     SkString    b((size_t)0);
     22     SkString    c("");
     23     SkString    d(nullptr, 0);
     24 
     25     REPORTER_ASSERT(reporter, a.isEmpty());
     26     REPORTER_ASSERT(reporter, a == b && a == c && a == d);
     27 
     28     a.set("hello");
     29     b.set("hellox", 5);
     30     c.set(a);
     31     d.resize(5);
     32     memcpy(d.writable_str(), "helloz", 5);
     33 
     34     REPORTER_ASSERT(reporter, !a.isEmpty());
     35     REPORTER_ASSERT(reporter, a.size() == 5);
     36     REPORTER_ASSERT(reporter, a == b && a == c && a == d);
     37     REPORTER_ASSERT(reporter, a.equals("hello", 5));
     38     REPORTER_ASSERT(reporter, a.equals("hello"));
     39     REPORTER_ASSERT(reporter, !a.equals("help"));
     40 
     41     REPORTER_ASSERT(reporter,  a.startsWith("hell"));
     42     REPORTER_ASSERT(reporter,  a.startsWith('h'));
     43     REPORTER_ASSERT(reporter, !a.startsWith( "ell"));
     44     REPORTER_ASSERT(reporter, !a.startsWith( 'e'));
     45     REPORTER_ASSERT(reporter,  a.startsWith(""));
     46     REPORTER_ASSERT(reporter,  a.endsWith("llo"));
     47     REPORTER_ASSERT(reporter,  a.endsWith('o'));
     48     REPORTER_ASSERT(reporter, !a.endsWith("ll" ));
     49     REPORTER_ASSERT(reporter, !a.endsWith('l'));
     50     REPORTER_ASSERT(reporter,  a.endsWith(""));
     51     REPORTER_ASSERT(reporter,  a.contains("he"));
     52     REPORTER_ASSERT(reporter,  a.contains("ll"));
     53     REPORTER_ASSERT(reporter,  a.contains("lo"));
     54     REPORTER_ASSERT(reporter,  a.contains("hello"));
     55     REPORTER_ASSERT(reporter, !a.contains("hellohello"));
     56     REPORTER_ASSERT(reporter,  a.contains(""));
     57     REPORTER_ASSERT(reporter,  a.contains('e'));
     58     REPORTER_ASSERT(reporter, !a.contains('z'));
     59 
     60     SkString    e(a);
     61     SkString    f("hello");
     62     SkString    g("helloz", 5);
     63 
     64     REPORTER_ASSERT(reporter, a == e && a == f && a == g);
     65 
     66     b.set("world");
     67     c = b;
     68     REPORTER_ASSERT(reporter, a != b && a != c && b == c);
     69 
     70     a.append(" world");
     71     e.append("worldz", 5);
     72     e.insert(5, " ");
     73     f.set("world");
     74     f.prepend("hello ");
     75     REPORTER_ASSERT(reporter, a.equals("hello world") && a == e && a == f);
     76 
     77     a.reset();
     78     b.resize(0);
     79     REPORTER_ASSERT(reporter, a.isEmpty() && b.isEmpty() && a == b);
     80 
     81     a.set("a");
     82     a.set("ab");
     83     a.set("abc");
     84     a.set("abcd");
     85 
     86     a.set("");
     87     a.appendS32(0x7FFFFFFFL);
     88     REPORTER_ASSERT(reporter, a.equals("2147483647"));
     89     a.set("");
     90     a.appendS32(0x80000001L);
     91     REPORTER_ASSERT(reporter, a.equals("-2147483647"));
     92     a.set("");
     93     a.appendS32(0x80000000L);
     94     REPORTER_ASSERT(reporter, a.equals("-2147483648"));
     95 
     96     a.set("");
     97     a.appendU32(0x7FFFFFFFUL);
     98     REPORTER_ASSERT(reporter, a.equals("2147483647"));
     99     a.set("");
    100     a.appendU32(0x80000001UL);
    101     REPORTER_ASSERT(reporter, a.equals("2147483649"));
    102     a.set("");
    103     a.appendU32(0xFFFFFFFFUL);
    104     REPORTER_ASSERT(reporter, a.equals("4294967295"));
    105 
    106     a.set("");
    107     a.appendS64(0x7FFFFFFFFFFFFFFFLL, 0);
    108     REPORTER_ASSERT(reporter, a.equals("9223372036854775807"));
    109     a.set("");
    110     a.appendS64(0x8000000000000001LL, 0);
    111     REPORTER_ASSERT(reporter, a.equals("-9223372036854775807"));
    112     a.set("");
    113     a.appendS64(0x8000000000000000LL, 0);
    114     REPORTER_ASSERT(reporter, a.equals("-9223372036854775808"));
    115     a.set("");
    116     a.appendS64(0x0000000001000000LL, 15);
    117     REPORTER_ASSERT(reporter, a.equals("000000016777216"));
    118     a.set("");
    119     a.appendS64(0xFFFFFFFFFF000000LL, 15);
    120     REPORTER_ASSERT(reporter, a.equals("-000000016777216"));
    121 
    122     a.set("");
    123     a.appendU64(0x7FFFFFFFFFFFFFFFULL, 0);
    124     REPORTER_ASSERT(reporter, a.equals("9223372036854775807"));
    125     a.set("");
    126     a.appendU64(0x8000000000000001ULL, 0);
    127     REPORTER_ASSERT(reporter, a.equals("9223372036854775809"));
    128     a.set("");
    129     a.appendU64(0xFFFFFFFFFFFFFFFFULL, 0);
    130     REPORTER_ASSERT(reporter, a.equals("18446744073709551615"));
    131     a.set("");
    132     a.appendU64(0x0000000001000000ULL, 15);
    133     REPORTER_ASSERT(reporter, a.equals("000000016777216"));
    134 
    135     a.printf("%i", 0);
    136     REPORTER_ASSERT(reporter, a.equals("0"));
    137     a.printf("%g", 3.14);
    138     REPORTER_ASSERT(reporter, a.equals("3.14"));
    139     a.printf("hello %s", "skia");
    140     REPORTER_ASSERT(reporter, a.equals("hello skia"));
    141 
    142     static const struct {
    143         SkScalar    fValue;
    144         const char* fString;
    145     } gRec[] = {
    146         { 0,            "0" },
    147         { SK_Scalar1,   "1" },
    148         { -SK_Scalar1,  "-1" },
    149         { SK_Scalar1/2, "0.5" },
    150   #if defined(SK_BUILD_FOR_WIN) && (_MSC_VER < 1900)
    151         { 3.4028234e38f,   "3.4028235e+038" },
    152         { -3.4028234e38f, "-3.4028235e+038" },
    153   #else
    154         { 3.4028234e38f,   "3.4028235e+38" },
    155         { -3.4028234e38f, "-3.4028235e+38" },
    156   #endif
    157     };
    158     for (size_t i = 0; i < SK_ARRAY_COUNT(gRec); i++) {
    159         a.reset();
    160         a.appendScalar(gRec[i].fValue);
    161         REPORTER_ASSERT(reporter, a.size() <= SkStrAppendScalar_MaxSize);
    162         if (!a.equals(gRec[i].fString)) {
    163             ERRORF(reporter, "received <%s> expected <%s>\n", a.c_str(), gRec[i].fString);
    164         }
    165     }
    166 
    167     REPORTER_ASSERT(reporter, SkStringPrintf("%i", 0).equals("0"));
    168 
    169     char buffer [40];
    170     memset(buffer, 'a', 40);
    171     REPORTER_ASSERT(reporter, buffer[18] == 'a');
    172     REPORTER_ASSERT(reporter, buffer[19] == 'a');
    173     REPORTER_ASSERT(reporter, buffer[20] == 'a');
    174     snprintf(buffer, 20, gThirtyWideDecimal, 0);
    175     REPORTER_ASSERT(reporter, buffer[18] == ' ');
    176     REPORTER_ASSERT(reporter, buffer[19] == 0);
    177     REPORTER_ASSERT(reporter, buffer[20] == 'a');
    178 
    179     REPORTER_ASSERT(reporter, SkStringPrintf("%i", 0).equals("0"));
    180 
    181     // 2000 is larger than the static buffer size inside SkString.cpp
    182     a = SkStringPrintf("%2000s", " ");
    183     REPORTER_ASSERT(reporter, a.size() == 2000);
    184     for (size_t i = 0; i < a.size(); ++i) {
    185         if (a[i] != ' ') {
    186             ERRORF(reporter, "SkStringPrintf fail: a[%d] = '%c'", i, a[i]);
    187             break;
    188         }
    189     }
    190     a.reset();
    191     a.printf("%2000s", " ");
    192     REPORTER_ASSERT(reporter, a.size() == 2000);
    193     for (size_t i = 0; i < a.size(); ++i) {
    194         if (a[i] != ' ') {
    195             ERRORF(reporter, "SkString::printf fail: a[%d] = '%c'", i, a[i]);
    196             break;
    197         }
    198     }
    199     a.appendf("%2000s", " ");
    200     REPORTER_ASSERT(reporter, a.size() == 4000);
    201     for (size_t i = 0; i < a.size(); ++i) {
    202         if (a[i] != ' ') {
    203             ERRORF(reporter, "SkString::appendf fail: a[%d] = '%c'", i, a[i]);
    204             break;
    205         }
    206     }
    207 }
    208 
    209 DEF_TEST(String_SkStrSplit, r) {
    210     SkTArray<SkString> results;
    211 
    212     SkStrSplit("a-_b_c-dee--f-_-_-g-", "-_", &results);
    213     REPORTER_ASSERT(r, results.count() == 6);
    214     REPORTER_ASSERT(r, results[0].equals("a"));
    215     REPORTER_ASSERT(r, results[1].equals("b"));
    216     REPORTER_ASSERT(r, results[2].equals("c"));
    217     REPORTER_ASSERT(r, results[3].equals("dee"));
    218     REPORTER_ASSERT(r, results[4].equals("f"));
    219     REPORTER_ASSERT(r, results[5].equals("g"));
    220 
    221     results.reset();
    222     SkStrSplit("\n", "\n", &results);
    223     REPORTER_ASSERT(r, results.count() == 0);
    224 
    225     results.reset();
    226     SkStrSplit("", "\n", &results);
    227     REPORTER_ASSERT(r, results.count() == 0);
    228 
    229     results.reset();
    230     SkStrSplit("a", "\n", &results);
    231     REPORTER_ASSERT(r, results.count() == 1);
    232     REPORTER_ASSERT(r, results[0].equals("a"));
    233 }
    234 DEF_TEST(String_SkStrSplit_All, r) {
    235     SkTArray<SkString> results;
    236     SkStrSplit("a-_b_c-dee--f-_-_-g-", "-_", kStrict_SkStrSplitMode, &results);
    237     REPORTER_ASSERT(r, results.count() == 13);
    238     REPORTER_ASSERT(r, results[0].equals("a"));
    239     REPORTER_ASSERT(r, results[1].equals(""));
    240     REPORTER_ASSERT(r, results[2].equals("b"));
    241     REPORTER_ASSERT(r, results[3].equals("c"));
    242     REPORTER_ASSERT(r, results[4].equals("dee"));
    243     REPORTER_ASSERT(r, results[5].equals(""));
    244     REPORTER_ASSERT(r, results[6].equals("f"));
    245     REPORTER_ASSERT(r, results[7].equals(""));
    246     REPORTER_ASSERT(r, results[8].equals(""));
    247     REPORTER_ASSERT(r, results[9].equals(""));
    248     REPORTER_ASSERT(r, results[10].equals(""));
    249     REPORTER_ASSERT(r, results[11].equals("g"));
    250     REPORTER_ASSERT(r, results[12].equals(""));
    251 
    252     results.reset();
    253     SkStrSplit("\n", "\n", kStrict_SkStrSplitMode, &results);
    254     REPORTER_ASSERT(r, results.count() == 2);
    255     REPORTER_ASSERT(r, results[0].equals(""));
    256     REPORTER_ASSERT(r, results[1].equals(""));
    257 
    258     results.reset();
    259     SkStrSplit("", "\n", kStrict_SkStrSplitMode, &results);
    260     REPORTER_ASSERT(r, results.count() == 0);
    261 
    262     results.reset();
    263     SkStrSplit("a", "\n", kStrict_SkStrSplitMode, &results);
    264     REPORTER_ASSERT(r, results.count() == 1);
    265     REPORTER_ASSERT(r, results[0].equals("a"));
    266 
    267     results.reset();
    268     SkStrSplit(",,", ",", kStrict_SkStrSplitMode, &results);
    269     REPORTER_ASSERT(r, results.count() == 3);
    270     REPORTER_ASSERT(r, results[0].equals(""));
    271     REPORTER_ASSERT(r, results[1].equals(""));
    272     REPORTER_ASSERT(r, results[2].equals(""));
    273 
    274     results.reset();
    275     SkStrSplit(",a,b,", ",", kStrict_SkStrSplitMode, &results);
    276     REPORTER_ASSERT(r, results.count() == 4);
    277     REPORTER_ASSERT(r, results[0].equals(""));
    278     REPORTER_ASSERT(r, results[1].equals("a"));
    279     REPORTER_ASSERT(r, results[2].equals("b"));
    280     REPORTER_ASSERT(r, results[3].equals(""));
    281 }
    282 
    283 // https://bugs.chromium.org/p/skia/issues/detail?id=7107
    284 DEF_TEST(String_Threaded, r) {
    285     SkString str("foo");
    286 
    287     std::thread threads[5];
    288     for (auto& thread : threads) {
    289         thread = std::thread([&] {
    290             SkString copy = str;
    291             (void)copy.equals("test");
    292         });
    293     }
    294     for (auto& thread : threads) {
    295         thread.join();
    296     }
    297 }
    298 
    299 // Ensure that the string allocate doesn't internally overflow any calculations, and accidentally
    300 // let us create a string with a requested length longer than we can manage.
    301 DEF_TEST(String_huge, r) {
    302     // start testing slightly below max 32
    303     size_t size = UINT32_MAX - 16;
    304     // See where we crash, and manually check that its at the right point.
    305     //
    306     //  To test, change the false to true
    307     while (false) {
    308         // On a 64bit build, this should crash when size == 1 << 32, since we can't store
    309         // that length in the string's header (which has a u32 slot for the length).
    310         //
    311         // On a 32bit build, this should crash the first time around, since we can't allocate
    312         // anywhere near this amount.
    313         //
    314         SkString str(size);
    315         size += 1;
    316     }
    317 }
    318 
    319 DEF_TEST(String_fromUTF16, r) {
    320     // test data produced with `iconv`.
    321     const uint16_t test1[] = {
    322         0xD835, 0xDCD0, 0xD835, 0xDCD1, 0xD835, 0xDCD2, 0xD835, 0xDCD3, 0xD835, 0xDCD4, 0x0020,
    323         0xD835, 0xDCD5, 0xD835, 0xDCD6, 0xD835, 0xDCD7, 0xD835, 0xDCD8, 0xD835, 0xDCD9
    324     };
    325     REPORTER_ASSERT(r, SkStringFromUTF16(test1, SK_ARRAY_COUNT(test1)).equals(" "));
    326 
    327     const uint16_t test2[] = {
    328         0x0041, 0x0042, 0x0043, 0x0044, 0x0045, 0x0020, 0x0046, 0x0047, 0x0048, 0x0049, 0x004A,
    329     };
    330     REPORTER_ASSERT(r, SkStringFromUTF16(test2, SK_ARRAY_COUNT(test2)).equals("ABCDE FGHIJ"));
    331 
    332     const uint16_t test3[] = {
    333         0x03B1, 0x03B2, 0x03B3, 0x03B4, 0x03B5, 0x0020, 0x03B6, 0x03B7, 0x03B8, 0x03B9, 0x03BA,
    334     };
    335     REPORTER_ASSERT(r, SkStringFromUTF16(test3, SK_ARRAY_COUNT(test3)).equals(" "));
    336 }
    337 
    338