/**************************************************************************** ** ** Copyright (C) 2015 The Qt Company Ltd. ** Contact: http://www.qt.io/licensing/ ** ** This file is part of the QtNfc module of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:LGPL21$ ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and The Qt Company. For licensing terms ** and conditions see http://www.qt.io/terms-conditions. For further ** information use the contact form at http://www.qt.io/contact-us. ** ** GNU Lesser General Public License Usage ** Alternatively, this file may be used under the terms of the GNU Lesser ** General Public License version 2.1 or version 3 as published by the Free ** Software Foundation and appearing in the file LICENSE.LGPLv21 and ** LICENSE.LGPLv3 included in the packaging of this file. Please review the ** following information to ensure the GNU Lesser General Public License ** requirements will be met: https://www.gnu.org/licenses/lgpl.html and ** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. ** ** As a special exception, The Qt Company gives you certain additional ** rights. These rights are described in The Qt Company LGPL Exception ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #include #include #include #include #include #include QT_USE_NAMESPACE Q_DECLARE_METATYPE(QNearFieldTarget*) static const char * const deadbeef = "\xde\xad\xbe\xef"; class tst_QNearFieldTagType2 : public QObject { Q_OBJECT public: tst_QNearFieldTagType2(); private slots: void init(); void cleanup(); void staticMemoryModel(); void dynamicMemoryModel(); void ndefMessages(); private: void waitForMatchingTarget(); QNearFieldManagerPrivateImpl *emulatorBackend; QNearFieldManager *manager; QNearFieldTagType2 *target; }; tst_QNearFieldTagType2::tst_QNearFieldTagType2() : emulatorBackend(0), manager(0), target(0) { QDir::setCurrent(QLatin1String(SRCDIR)); qRegisterMetaType(); qRegisterMetaType(); } void tst_QNearFieldTagType2::init() { emulatorBackend = new QNearFieldManagerPrivateImpl; manager = new QNearFieldManager(emulatorBackend, 0); } void tst_QNearFieldTagType2::cleanup() { emulatorBackend->reset(); delete manager; manager = 0; emulatorBackend = 0; target = 0; } void tst_QNearFieldTagType2::waitForMatchingTarget() { QSignalSpy targetDetectedSpy(manager, SIGNAL(targetDetected(QNearFieldTarget*))); manager->startTargetDetection(); QTRY_VERIFY(!targetDetectedSpy.isEmpty()); target = qobject_cast(targetDetectedSpy.first().at(0).value()); manager->stopTargetDetection(); QVERIFY(target); QCOMPARE(target->type(), QNearFieldTarget::NfcTagType2); } void tst_QNearFieldTagType2::staticMemoryModel() { waitForMatchingTarget(); QVERIFY(target->accessMethods() & QNearFieldTarget::TagTypeSpecificAccess); QCOMPARE(target->version(), quint8(0x10)); // readBlock(), writeBlock() { for (int i = 0; i < 2; ++i) { QNearFieldTarget::RequestId id = target->readBlock(i); QVERIFY(target->waitForRequestCompleted(id)); const QByteArray block = target->requestResponse(id).toByteArray(); id = target->writeBlock(i, QByteArray(4, 0x55)); QVERIFY(target->waitForRequestCompleted(id)); QVERIFY(!target->requestResponse(id).toBool()); id = target->readBlock(i); QVERIFY(target->waitForRequestCompleted(id)); const QByteArray readBlock = target->requestResponse(id).toByteArray(); QCOMPARE(readBlock, block); } for (int i = 3; i < 16; ++i) { // Read initial data QNearFieldTarget::RequestId id = target->readBlock(i); QVERIFY(target->waitForRequestCompleted(id)); QByteArray initialBlock = target->requestResponse(id).toByteArray(); // Write 0x55 id = target->writeBlock(i, QByteArray(4, 0x55)); QVERIFY(target->waitForRequestCompleted(id)); QVERIFY(target->requestResponse(id).toBool()); id = target->readBlock(i); QVERIFY(target->waitForRequestCompleted(id)); QByteArray readBlock = target->requestResponse(id).toByteArray(); QCOMPARE(readBlock, QByteArray(4, 0x55) + initialBlock.mid(4)); // Write 0xaa id = target->writeBlock(i, QByteArray(4, char(0xaa))); QVERIFY(target->waitForRequestCompleted(id)); QVERIFY(target->requestResponse(id).toBool()); id = target->readBlock(i); QVERIFY(target->waitForRequestCompleted(id)); readBlock = target->requestResponse(id).toByteArray(); QCOMPARE(readBlock, QByteArray(4, char(0xaa)) + initialBlock.mid(4)); } } } void tst_QNearFieldTagType2::dynamicMemoryModel() { bool testedStatic = false; bool testedDynamic = false; QList seenIds; forever { waitForMatchingTarget(); QVERIFY(target->accessMethods() & QNearFieldTarget::TagTypeSpecificAccess); QNearFieldTarget::RequestId id = target->readBlock(0); QVERIFY(target->waitForRequestCompleted(id)); const QByteArray data = target->requestResponse(id).toByteArray(); const QByteArray uid = data.left(3) + data.mid(4, 4); if (seenIds.contains(uid)) break; else seenIds.append(uid); QCOMPARE(target->version(), quint8(0x10)); bool dynamic = target->memorySize() > 1024; if (dynamic) { testedDynamic = true; int totalBlocks = target->memorySize() / 4; int sector1Blocks = qMin(totalBlocks - 256, 256); // default sector is sector 0 for (int i = 3; i < 256; ++i) { // Write 0x55 QNearFieldTarget::RequestId id = target->writeBlock(i, deadbeef); QVERIFY(target->waitForRequestCompleted(id)); QVERIFY(target->requestResponse(id).toBool()); } // change to sector 1 { QNearFieldTarget::RequestId id = target->selectSector(1); QVERIFY(target->waitForRequestCompleted(id)); QVERIFY(target->requestResponse(id).toBool()); } for (int i = 0; i < sector1Blocks; ++i) { QNearFieldTarget::RequestId id = target->readBlock(i); QVERIFY(target->waitForRequestCompleted(id)); QByteArray initialBlock = target->requestResponse(id).toByteArray(); QVERIFY(initialBlock.left(4) != deadbeef); // Write 0x55 id = target->writeBlock(i, QByteArray(4, 0x55)); QVERIFY(target->waitForRequestCompleted(id)); QVERIFY(target->requestResponse(id).toBool()); id = target->readBlock(i); QVERIFY(target->waitForRequestCompleted(id)); QByteArray readBlock = target->requestResponse(id).toByteArray(); QCOMPARE(readBlock, QByteArray(4, 0x55) + initialBlock.mid(4)); // Write 0xaa id = target->writeBlock(i, QByteArray(4, char(0xaa))); QVERIFY(target->waitForRequestCompleted(id)); QVERIFY(target->requestResponse(id).toBool()); id = target->readBlock(i); QVERIFY(target->waitForRequestCompleted(id)); readBlock = target->requestResponse(id).toByteArray(); QCOMPARE(readBlock, QByteArray(4, char(0xaa)) + initialBlock.mid(4)); } // change to sector 0 { QNearFieldTarget::RequestId id = target->selectSector(0); QVERIFY(target->waitForRequestCompleted(id)); QVERIFY(target->requestResponse(id).toBool()); } for (int i = 3; i < 256; ++i) { QNearFieldTarget::RequestId id = target->readBlock(i); QVERIFY(target->waitForRequestCompleted(id)); QByteArray readBlock = target->requestResponse(id).toByteArray(); QVERIFY(readBlock.left(4) == deadbeef); } } else { testedStatic = true; QNearFieldTarget::RequestId id = target->selectSector(1); QVERIFY(target->waitForRequestCompleted(id)); QVERIFY(!target->requestResponse(id).toBool()); } } QVERIFY(testedStatic); QVERIFY(testedDynamic); } void tst_QNearFieldTagType2::ndefMessages() { QSKIP("Not implemented"); QByteArray firstId; forever { waitForMatchingTarget(); QNearFieldTarget::RequestId id = target->readBlock(0); QVERIFY(target->waitForRequestCompleted(id)); QByteArray uid = target->requestResponse(id).toByteArray().left(3); id = target->readBlock(1); QVERIFY(target->waitForRequestCompleted(id)); uid.append(target->requestResponse(id).toByteArray()); if (firstId.isEmpty()) firstId = uid; else if (firstId == uid) break; QVERIFY(target->hasNdefMessage()); QSignalSpy ndefMessageReadSpy(target, SIGNAL(ndefMessageRead(QNdefMessage))); target->readNdefMessages(); QTRY_VERIFY(!ndefMessageReadSpy.isEmpty()); QList ndefMessages; for (int i = 0; i < ndefMessageReadSpy.count(); ++i) ndefMessages.append(ndefMessageReadSpy.at(i).first().value()); QList messages; QNdefNfcTextRecord textRecord; textRecord.setText(QStringLiteral("tst_QNearFieldTagType2::ndefMessages")); QNdefMessage message; message.append(textRecord); if (target->memorySize() > 120) { QNdefRecord record; record.setTypeNameFormat(QNdefRecord::ExternalRtd); record.setType("org.qt-project:ndefMessagesTest"); record.setPayload(QByteArray(120, quint8(0x55))); message.append(record); } messages.append(message); QSignalSpy ndefMessageWriteSpy(target, SIGNAL(ndefMessagesWritten())); target->writeNdefMessages(messages); QTRY_VERIFY(!ndefMessageWriteSpy.isEmpty()); QVERIFY(target->hasNdefMessage()); ndefMessageReadSpy.clear(); target->readNdefMessages(); QTRY_VERIFY(!ndefMessageReadSpy.isEmpty()); QList storedMessages; for (int i = 0; i < ndefMessageReadSpy.count(); ++i) storedMessages.append(ndefMessageReadSpy.at(i).first().value()); QVERIFY(ndefMessages != storedMessages); QVERIFY(messages == storedMessages); } } QTEST_MAIN(tst_QNearFieldTagType2) // Unset the moc namespace which is not required for the following include. #undef QT_BEGIN_MOC_NAMESPACE #define QT_BEGIN_MOC_NAMESPACE #undef QT_END_MOC_NAMESPACE #define QT_END_MOC_NAMESPACE #include "tst_qnearfieldtagtype2.moc"