aboutsummaryrefslogtreecommitdiffstats
path: root/sources/shiboken6/generator/shiboken/generatorargument.cpp
blob: e81ad0797ed7d9e3075a0784f9acfeca19bc1838 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
// Copyright (C) 2022 The Qt Company Ltd.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0

#include "generatorargument.h"
#include <abstractmetatype.h>
#include <messages.h>
#include <typesystem.h>

#include <QtCore/QDebug>
#include <QtCore/QSet>

static bool isCppPrimitiveString(const AbstractMetaType &type)
{
    return type.referenceType() == NoReference && type.indirections() == 1
        && AbstractMetaType::cppSignedCharTypes().contains(type.name());
}

GeneratorArgument GeneratorArgument::fromMetaType(const AbstractMetaType &type)
{
    GeneratorArgument result;

    const auto typeEntry = type.typeEntry();
    if (typeEntry->isCustom() || typeEntry->isVarargs())
        return result;

    result.indirections = -type.indirectionsV().size();
    if (isCppPrimitiveString(type)
        || type.isVoidPointer()
        || type.typeUsagePattern() == AbstractMetaType::NativePointerAsArrayPattern) {
        result.indirections += 1;
    }

    if (typeEntry->isEnum()) {
        result.type = Type::Enum;
    } else if (typeEntry->isFlags()) {
        result.type = Type::Flags;
    } else if (typeEntry->isContainer()) {
        result.type = Type::Container;
    } else {
        if (typeEntry->isPrimitive())
            result.type = Type::Primitive;

        const AbstractMetaTypeList &nestedArrayTypes = type.nestedArrayTypes();
        if (!nestedArrayTypes.isEmpty()) {
            if (nestedArrayTypes.constLast().isCppPrimitive()) {
                result.type = Type::CppPrimitiveArray;
            } else {
                static QSet<QString> warnedTypes;
                const QString &signature = type.cppSignature();
                if (!warnedTypes.contains(signature)) {
                    warnedTypes.insert(signature);
                    qWarning("%s", qPrintable(msgUnknownArrayPointerConversion(signature)));
                }
                result.indirections -= 1;
            }
        }
    }

    if (result.type == Type::Other || result.type == Type::Primitive) {
        if (type.valueTypeWithCopyConstructorOnlyPassed()) {
            result.flags.setFlag(Flag::TreatAsPointer);
        } else if ((type.isObjectType() || type.isPointer())
                   && !type.isUserPrimitive() && !type.isExtendedCppPrimitive()) {
            result.flags.setFlag(Flag::PointerOrObjectType);
        } else if (type.referenceType() == LValueReference
                   && !type.isUserPrimitive()
                   && !type.isExtendedCppPrimitive()) {
            result.flags.setFlag(Flag::MayHaveImplicitConversion);
        }
    }

    // For implicit conversions or containers, either value or pointer conversion
    // may occur. An implicit conversion uses value conversion whereas the object
    // itself uses pointer conversion. For containers, the PyList/container
    // conversion is by value whereas opaque containers use pointer conversion.
    // For a container passed by pointer, a local variable is also needed.
    if (result.flags.testFlag(Flag::MayHaveImplicitConversion)
        || type.generateOpaqueContainer()
        || (result.type == Type::Container && result.indirections != 0)) {
        result.flags.setFlag(Flag::ValueOrPointer);
    }

    if (result.type == Type::CppPrimitiveArray) {
        result.conversion = Conversion::CppPrimitiveArray;
    } else if (result.flags.testFlag(Flag::ValueOrPointer)) {
        result.conversion = Conversion::ValueOrPointer;
        ++result.indirections;
    } else if (result.flags.testAnyFlags(Flag::TreatAsPointer | Flag::PointerOrObjectType)) {
        result.conversion = Conversion::Pointer;
        ++result.indirections;
    }

    return result;
}

QDebug operator<<(QDebug debug, const GeneratorArgument &a)
{
    QDebugStateSaver saver(debug);
    debug.noquote();
    debug.nospace();
    debug << "GeneratorArgument(" << a.type;
    if (a.conversion != GeneratorArgument::Conversion::Default)
        debug << ", conversion=" << a.conversion;
    if (a.flags)
        debug << ", flags(" << a.flags;
    if (a.indirections != 0)
        debug << ", indirections=" << a.indirections;
    debug << ')';
    return debug;
}