/* * This file is part of the API Extractor project. * * Copyright (C) 2009 Nokia Corporation and/or its subsidiary(-ies). * * Contact: PySide team * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * version 2 as published by the Free Software Foundation. * * This program is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA * 02110-1301 USA * */ #include "testtemplates.h" #include #include #include "testutil.h" void TestTemplates::testTemplateWithNamespace() { const char cppCode[] = "\ struct Url {\ void name();\ };\ namespace Internet {\ struct Url{};\ struct Bookmarks {\ QList list();\ };\ }"; const char xmlCode0[] = "\ \ \ "; QTemporaryFile file; QVERIFY(file.open()); file.write(xmlCode0); file.close(); QString xmlCode1 = QString("\ \ \ \ \ \ \ ").arg(file.fileName()); TestUtil t(cppCode, qPrintable(xmlCode1), false); AbstractMetaClassList classes = t.builder()->classes(); AbstractMetaClass* classB = classes.findClass("Bookmarks"); QVERIFY(classB); const AbstractMetaFunction* func = classB->findFunction("list"); AbstractMetaType* funcType = func->type(); QVERIFY(funcType); QCOMPARE(funcType->cppSignature(), QString("QList")); } void TestTemplates::testTemplateOnContainers() { const char cppCode[] = "\ struct Base {};\ namespace Namespace {\ enum SomeEnum { E1, E2 };\ template struct A {\ A foo(const QList >& a);\ };\ typedef A B;\ }\ "; const char xmlCode[] = "\ \ \ \ \ \ \ \ "; TestUtil t(cppCode, xmlCode, false); AbstractMetaClassList classes = t.builder()->classes(); AbstractMetaClass* classB = classes.findClass("B"); QVERIFY(!classB->baseClass()); QVERIFY(classB->baseClassName().isNull()); const AbstractMetaFunction* func = classB->findFunction("foo"); AbstractMetaType* argType = func->arguments().first()->type(); QCOMPARE(argType->instantiations().count(), 1); QCOMPARE(argType->typeEntry()->qualifiedCppName(), QString("QList")); const AbstractMetaType* instance1 = argType->instantiations().first(); QCOMPARE(instance1->instantiations().count(), 1); QCOMPARE(instance1->typeEntry()->qualifiedCppName(), QString("Namespace::A")); const AbstractMetaType* instance2 = instance1->instantiations().first(); QCOMPARE(instance2->instantiations().count(), 0); QCOMPARE(instance2->typeEntry()->qualifiedCppName(), QString("Namespace::E1")); } void TestTemplates::testTemplateValueAsArgument() { const char cppCode[] = "\ template struct List() {};\ void func(List arg) {}\ "; const char xmlCode[] = "\ \ \ \ \ \ "; TestUtil t(cppCode, xmlCode, false); AbstractMetaFunctionList globalFuncs = t.builder()->globalFunctions(); QCOMPARE(globalFuncs.count(), 1); AbstractMetaFunction* func = globalFuncs.first(); QCOMPARE(func->minimalSignature(), QString("func(List)")); QCOMPARE(func->arguments().first()->type()->cppSignature(), QString("List")); } void TestTemplates::testTemplatePointerAsArgument() { const char cppCode[] = "\ template struct List() {};\ void func(List* arg) {}\ "; const char xmlCode[] = "\ \ \ \ \ \ "; TestUtil t(cppCode, xmlCode, false); AbstractMetaFunctionList globalFuncs = t.builder()->globalFunctions(); QCOMPARE(globalFuncs.count(), 1); AbstractMetaFunction* func = globalFuncs.first(); QCOMPARE(func->minimalSignature(), QString("func(List*)")); QCOMPARE(func->arguments().first()->type()->cppSignature(), QString("List *")); } void TestTemplates::testTemplateReferenceAsArgument() { const char cppCode[] = "\ template struct List() {};\ void func(List& arg) {}\ "; const char xmlCode[] = "\ \ \ \ \ \ "; TestUtil t(cppCode, xmlCode, false); AbstractMetaFunctionList globalFuncs = t.builder()->globalFunctions(); QCOMPARE(globalFuncs.count(), 1); AbstractMetaFunction* func = globalFuncs.first(); QCOMPARE(func->minimalSignature(), QString("func(List&)")); QCOMPARE(func->arguments().first()->type()->cppSignature(), QString("List &")); } void TestTemplates::testInheritanceFromContainterTemplate() { const char cppCode[] = "\ template\ struct ListContainer {\ inline void push_front(const T& t);\ inline T& front();\ };\ struct FooBar {};\ struct FooBars : public ListContainer {};\ "; const char xmlCode[] = "\ \ \ \ \ \ \ \ \ "; TestUtil t(cppCode, xmlCode, false); AbstractMetaClassList classes = t.builder()->classes(); AbstractMetaClassList templates = t.builder()->templates(); QCOMPARE(classes.count(), 2); QCOMPARE(templates.count(), 1); const AbstractMetaClass* foobars = classes.findClass("FooBars"); QCOMPARE(foobars->functions().count(), 4); const AbstractMetaClass* lc = templates.first(); QCOMPARE(lc->functions().count(), 2); } void TestTemplates::testTemplateInheritanceMixedWithForwardDeclaration() { const char cppCode[] = "\ enum SomeEnum { E1, E2 };\ template struct Future;\ template\ struct A {\ A();\ void method();\ friend struct Future;\ };\ typedef A B;\ template struct Future {};\ "; const char xmlCode[] = "\ \ \ \ \ \ "; TestUtil t(cppCode, xmlCode, false); AbstractMetaClassList classes = t.builder()->classes(); AbstractMetaClass* classB = classes.findClass("B"); QVERIFY(!classB->baseClass()); QVERIFY(classB->baseClassName().isNull()); // 3 functions: simple constructor, copy constructor and "method()". QCOMPARE(classB->functions().count(), 3); } void TestTemplates::testTemplateInheritanceMixedWithNamespaceAndForwardDeclaration() { const char cppCode[] = "\ namespace Namespace {\ enum SomeEnum { E1, E2 };\ template struct Future;\ template\ struct A {\ A();\ void method();\ friend struct Future;\ };\ typedef A B;\ template struct Future {};\ };\ "; const char xmlCode[] = "\ \ \ \ \ \ \ "; TestUtil t(cppCode, xmlCode, false); AbstractMetaClassList classes = t.builder()->classes(); AbstractMetaClass* classB = classes.findClass("Namespace::B"); QVERIFY(!classB->baseClass()); QVERIFY(classB->baseClassName().isNull()); // 3 functions: simple constructor, copy constructor and "method()". QCOMPARE(classB->functions().count(), 3); } void TestTemplates::testTypedefOfInstantiationOfTemplateClass() { const char cppCode[] = "\ namespace NSpace {\ enum ClassType {\ TypeOne\ };\ template\ struct BaseTemplateClass {\ inline ClassType getClassType() const { CLASS_TYPE; }\ };\ typedef BaseTemplateClass TypeOneClass;\ }\ "; const char xmlCode[] = "\ \ \ \ \ \ \ \ "; TestUtil t(cppCode, xmlCode, false); AbstractMetaClassList classes = t.builder()->classes(); QCOMPARE(classes.count(), 3); const AbstractMetaClass* base = classes.findClass("BaseTemplateClass"); QVERIFY(base); const AbstractMetaClass* one = classes.findClass("TypeOneClass"); QVERIFY(one); QCOMPARE(one->templateBaseClass(), base); QCOMPARE(one->functions().count(), base->functions().count()); QVERIFY(one->isTypeAlias()); const ComplexTypeEntry* oneType = one->typeEntry(); const ComplexTypeEntry* baseType = base->typeEntry(); QCOMPARE(oneType->baseContainerType(), baseType); QCOMPARE(one->baseClassNames(), QStringList("BaseTemplateClass")); QVERIFY(one->hasTemplateBaseClassInstantiations()); AbstractMetaTypeList instantiations = one->templateBaseClassInstantiations(); QCOMPARE(instantiations.count(), 1); const AbstractMetaType* inst = instantiations.first(); QVERIFY(inst); QVERIFY(!inst->isEnum()); QVERIFY(!inst->typeEntry()->isEnum()); QVERIFY(inst->typeEntry()->isEnumValue()); QCOMPARE(inst->cppSignature(), QString("NSpace::TypeOne")); } void TestTemplates::testContainerTypeIncompleteArgument() { const char* cppCode ="\ template\ class Vector {\ void method(const Vector& vector);\ Vector otherMethod();\ };\ template \ void Vector::method(const Vector& vector) {}\ Vector Vector::otherMethod() { return Vector(); }\ typedef Vector IntVector;\ "; const char* xmlCode = "\ \ \ \ \ "; TestUtil t(cppCode, xmlCode, true); AbstractMetaClassList classes = t.builder()->classes(); QCOMPARE(classes.count(), 1); AbstractMetaClass* vector = classes.findClass("IntVector"); QVERIFY(vector); QVERIFY(vector->typeEntry()->baseContainerType()); QCOMPARE(reinterpret_cast(vector->typeEntry()->baseContainerType())->type(), ContainerTypeEntry::VectorContainer); QCOMPARE(vector->functions().count(), 4); const AbstractMetaFunction* method = vector->findFunction("method"); QVERIFY(method); QCOMPARE(method->signature(), QString("method(const Vector & vector)")); const AbstractMetaFunction* otherMethod = vector->findFunction("otherMethod"); QVERIFY(otherMethod); QCOMPARE(otherMethod->signature(), QString("otherMethod()")); QVERIFY(otherMethod->type()); QCOMPARE(otherMethod->type()->cppSignature(), QString("Vector")); } QTEST_APPLESS_MAIN(TestTemplates) #include "testtemplates.moc"