blob: 93bae4afc1cc75ecd50a08ed9a891d621f7ad1e8 [file] [log] [blame]
#include <QtCore/QScopedPointer>
#include <QtTest/QTest>
#include <poppler-private.h>
#include <cstring>
#include <cstdint> // for uint16_t
#include "GlobalParams.h"
#include "UnicodeTypeTable.h"
#include "UTF.h"
class TestUTFConversion : public QObject
{
Q_OBJECT
public:
explicit TestUTFConversion(QObject *parent = nullptr) : QObject(parent) { }
private slots:
void testUTF_data();
void testUTF();
void testUnicodeToAscii7();
void testUnicodeLittleEndian();
};
static bool compare(const char *a, const char *b)
{
return strcmp(a, b) == 0;
}
static bool compare(const uint16_t *a, const uint16_t *b)
{
while (*a && *b) {
if (*a++ != *b++) {
return false;
}
}
return *a == *b;
}
static bool compare(const Unicode *a, const char *b, int len)
{
for (int i = 0; i < len; i++) {
if (a[i] != (Unicode)b[i]) {
return false;
}
}
return true;
}
static bool compare(const Unicode *a, const uint16_t *b, int len)
{
for (int i = 0; i < len; i++) {
if (a[i] != b[i]) {
return false;
}
}
return true;
}
void TestUTFConversion::testUTF_data()
{
QTest::addColumn<QString>("s");
QTest::newRow("<empty>") << QString(QLatin1String(""));
QTest::newRow("a") << QStringLiteral("a");
QTest::newRow("abc") << QStringLiteral("abc");
QTest::newRow("Latin") << QStringLiteral("Vitrum edere possum; mihi non nocet");
QTest::newRow("Greek") << QStringLiteral("Μπορώ να φάω σπασμένα γυαλιά χωρίς να πάθω τίποτα");
QTest::newRow("Icelandic") << QStringLiteral("Ég get etið gler án þess að meiða mig");
QTest::newRow("Russian") << QStringLiteral("Я могу есть стекло, оно мне не вредит.");
QTest::newRow("Sanskrit") << QStringLiteral("काचं शक्नोम्यत्तुम् । नोपहिनस्ति माम् ॥");
QTest::newRow("Arabic") << QStringLiteral("أنا قادر على أكل الزجاج و هذا لا يؤلمني");
QTest::newRow("Chinese") << QStringLiteral("我能吞下玻璃而不伤身体。");
QTest::newRow("Thai") << QStringLiteral("ฉันกินกระจกได้ แต่มันไม่ทำให้ฉันเจ็บ");
QTest::newRow("non BMP") << QStringLiteral("𝓹𝓸𝓹𝓹𝓵𝓮𝓻");
}
void TestUTFConversion::testUTF()
{
char utf8Buf[1000];
char *utf8String;
uint16_t utf16Buf[1000];
uint16_t *utf16String;
int len;
QFETCH(QString, s);
char *str = strdup(s.toUtf8().constData());
// UTF-8 to UTF-16
len = utf8CountUtf16CodeUnits(str);
QCOMPARE(len, s.size()); // QString size() returns number of code units, not code points
Q_ASSERT(len < (int)sizeof(utf16Buf)); // if this fails, make utf16Buf larger
len = utf8ToUtf16(str, utf16Buf, sizeof(utf16Buf), INT_MAX);
QVERIFY(compare(utf16Buf, s.utf16()));
QCOMPARE(len, s.size());
utf16String = utf8ToUtf16(str);
QVERIFY(compare(utf16String, s.utf16()));
free(utf16String);
std::string sUtf8(str);
std::string gsUtf16_a(utf8ToUtf16WithBom(sUtf8));
std::unique_ptr<GooString> gsUtf16_b(Poppler::QStringToUnicodeGooString(s));
QCOMPARE(gsUtf16_b->cmp(gsUtf16_a), 0);
// UTF-16 to UTF-8
len = utf16CountUtf8Bytes(s.utf16());
QCOMPARE(len, (int)strlen(str));
Q_ASSERT(len < (int)sizeof(utf8Buf)); // if this fails, make utf8Buf larger
len = utf16ToUtf8(s.utf16(), utf8Buf);
QVERIFY(compare(utf8Buf, str));
QCOMPARE(len, (int)strlen(str));
utf8String = utf16ToUtf8(s.utf16());
QVERIFY(compare(utf8String, str));
free(utf8String);
free(str);
}
void TestUTFConversion::testUnicodeToAscii7()
{
globalParams = std::make_unique<GlobalParams>();
// Test string is one 'Registered' and twenty 'Copyright' chars
// so it's long enough to reproduce the bug given that glibc
// malloc() always returns 8-byte aligned memory addresses.
GooString *goo = Poppler::QStringToUnicodeGooString(QString::fromUtf8("®©©©©©©©©©©©©©©©©©©©©")); // clazy:exclude=qstring-allocations
const std::vector<Unicode> in = TextStringToUCS4(goo->toStr());
delete goo;
int in_norm_len;
int *in_norm_idx;
Unicode *in_norm = unicodeNormalizeNFKC(in.data(), in.size(), &in_norm_len, &in_norm_idx, true);
Unicode *out;
int out_len;
int *out_ascii_idx;
unicodeToAscii7(std::span(in_norm, in_norm_len), &out, &out_len, in_norm_idx, &out_ascii_idx);
free(in_norm);
free(in_norm_idx);
// ascii7 conversion: ® -> (R) © -> (c)
const char *expected_ascii = (char *)"(R)(c)(c)(c)(c)(c)(c)(c)(c)(c)(c)(c)(c)(c)(c)(c)(c)(c)(c)(c)(c)";
QCOMPARE(out_len, (int)strlen(expected_ascii));
QVERIFY(compare(out, expected_ascii, out_len));
free(out);
free(out_ascii_idx);
}
void TestUTFConversion::testUnicodeLittleEndian()
{
uint16_t UTF16LE_hi[5] { 0xFFFE, 0x4800, 0x4900, 0x2100, 0x1126 }; // UTF16-LE "HI!☑"
std::string GooUTF16LE(reinterpret_cast<const char *>(UTF16LE_hi), sizeof(UTF16LE_hi));
uint16_t UTF16BE_hi[5] { 0xFEFF, 0x0048, 0x0049, 0x0021, 0x2611 }; // UTF16-BE "HI!☑"
std::string GooUTF16BE(reinterpret_cast<const char *>(UTF16BE_hi), sizeof(UTF16BE_hi));
// Let's assert both GooString's are different
QVERIFY(GooUTF16LE != GooUTF16BE);
const std::vector<Unicode> UCS4fromLE = TextStringToUCS4(GooUTF16LE);
const std::vector<Unicode> UCS4fromBE = TextStringToUCS4(GooUTF16BE);
// len is 4 because TextStringToUCS4() removes the two leading Byte Order Mark (BOM) code points
QCOMPARE(UCS4fromLE.size(), UCS4fromBE.size());
QCOMPARE(UCS4fromLE.size(), 4);
// Check that now after conversion, UCS4fromLE and UCS4fromBE are now the same
for (size_t i = 0; i < UCS4fromLE.size(); i++) {
QCOMPARE(UCS4fromLE[i], UCS4fromBE[i]);
}
const QString expected = QString::fromUtf8("HI!☑"); // clazy:exclude=qstring-allocations
// Do some final verifications, checking the strings to be "HI!"
QVERIFY(UCS4fromLE == UCS4fromBE);
QVERIFY(compare(UCS4fromLE.data(), expected.utf16(), UCS4fromLE.size()));
QVERIFY(compare(UCS4fromBE.data(), expected.utf16(), UCS4fromLE.size()));
}
QTEST_GUILESS_MAIN(TestUTFConversion)
#include "check_utf_conversion.moc"