From 3a726628f13c8f46b447cf0844eb8a5b740a1993 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Christian=20Str=C3=B8mme?= Date: Wed, 6 May 2015 18:15:03 +0200 Subject: [PATCH 01/24] Android: Store and use the class names as key when caching. Previously the jclass handle was part of the key used for caching the class' methods and fields. Using the jclass handle is not ideal, but it meant that we could easily create a key when the only identifier we had was the jobject or jclass handle. However, in Android 5.1, the re-use of handles seems to be more aggressive and therefore increasing the chance of a collision in the cache look-up. This change removes caching for all calls where we don't know the class name, as that is the only thing that guarantees that we create unique keys for each class. The consequence of this is that only calls that provide a class name will benefit from the internal caching. Task-number: QTBUG-45748 Change-Id: I0039d04e7c068debc9e3b3983632c45dc8e52309 Reviewed-by: Frank Meerkoetter Reviewed-by: Alex Blasche --- src/corelib/kernel/qjni.cpp | 472 ++++++++++++++++++++++-------------- src/corelib/kernel/qjni_p.h | 1 + 2 files changed, 293 insertions(+), 180 deletions(-) diff --git a/src/corelib/kernel/qjni.cpp b/src/corelib/kernel/qjni.cpp index 8431ee3b67..344f7725a0 100644 --- a/src/corelib/kernel/qjni.cpp +++ b/src/corelib/kernel/qjni.cpp @@ -43,7 +43,7 @@ QT_BEGIN_NAMESPACE static inline QString keyBase() { - return QStringLiteral("%1%2%3"); + return QStringLiteral("%1%2:%3"); } static QString qt_convertJString(jstring string) @@ -72,15 +72,15 @@ typedef QHash JClassHash; Q_GLOBAL_STATIC(JClassHash, cachedClasses) Q_GLOBAL_STATIC(QReadWriteLock, cachedClassesLock) -static QString toDotEncodedClassName(const char *className) +static QByteArray toBinaryEncClassName(const QByteArray &className) { - return QString::fromLatin1(className).replace(QLatin1Char('/'), QLatin1Char('.')); + return QByteArray(className).replace('/', '.'); } -static jclass getCachedClass(const QString &classDotEnc, bool *isCached = 0) +static jclass getCachedClass(const QByteArray &classBinEnc, bool *isCached = 0) { QReadLocker locker(cachedClassesLock); - const QHash::const_iterator &it = cachedClasses->constFind(classDotEnc); + const QHash::const_iterator &it = cachedClasses->constFind(QString::fromLatin1(classBinEnc)); const bool found = (it != cachedClasses->constEnd()); if (isCached != 0) @@ -89,10 +89,12 @@ static jclass getCachedClass(const QString &classDotEnc, bool *isCached = 0) return found ? it.value() : 0; } -static jclass loadClassDotEnc(const QString &classDotEnc, JNIEnv *env) +inline static jclass loadClass(const QByteArray &className, JNIEnv *env, bool binEncoded = false) { + const QByteArray &binEncClassName = binEncoded ? className : toBinaryEncClassName(className); + bool isCached = false; - jclass clazz = getCachedClass(classDotEnc, &isCached); + jclass clazz = getCachedClass(binEncClassName, &isCached); if (clazz != 0 || isCached) return clazz; @@ -102,11 +104,12 @@ static jclass loadClassDotEnc(const QString &classDotEnc, JNIEnv *env) QWriteLocker locker(cachedClassesLock); // did we lose the race? - const QHash::const_iterator &it = cachedClasses->constFind(classDotEnc); + const QLatin1String key(binEncClassName); + const QHash::const_iterator &it = cachedClasses->constFind(key); if (it != cachedClasses->constEnd()) return it.value(); - QJNIObjectPrivate stringName = QJNIObjectPrivate::fromString(classDotEnc); + QJNIObjectPrivate stringName = QJNIObjectPrivate::fromString(key); QJNIObjectPrivate classObject = classLoader.callObjectMethod("loadClass", "(Ljava/lang/String;)Ljava/lang/Class;", stringName.object()); @@ -114,27 +117,40 @@ static jclass loadClassDotEnc(const QString &classDotEnc, JNIEnv *env) if (!exceptionCheckAndClear(env) && classObject.isValid()) clazz = static_cast(env->NewGlobalRef(classObject.object())); - cachedClasses->insert(classDotEnc, clazz); + cachedClasses->insert(key, clazz); return clazz; } -inline static jclass loadClass(const char *className, JNIEnv *env) -{ - return loadClassDotEnc(toDotEncodedClassName(className), env); -} - typedef QHash JMethodIDHash; Q_GLOBAL_STATIC(JMethodIDHash, cachedMethodID) Q_GLOBAL_STATIC(QReadWriteLock, cachedMethodIDLock) +static inline jmethodID getMethodID(JNIEnv *env, + jclass clazz, + const char *name, + const char *sig, + bool isStatic = false) +{ + jmethodID id = isStatic ? env->GetStaticMethodID(clazz, name, sig) + : env->GetMethodID(clazz, name, sig); + + if (exceptionCheckAndClear(env)) + return 0; + + return id; +} + static jmethodID getCachedMethodID(JNIEnv *env, jclass clazz, + const QByteArray &className, const char *name, const char *sig, bool isStatic = false) { - // TODO: We need to use something else then the ref. from clazz to avoid collisions. - const QString key = keyBase().arg(size_t(clazz)).arg(QLatin1String(name)).arg(QLatin1String(sig)); + if (className.isEmpty()) + return getMethodID(env, clazz, name, sig, isStatic); + + const QString key = keyBase().arg(QLatin1String(className)).arg(QLatin1String(name)).arg(QLatin1String(sig)); QHash::const_iterator it; { @@ -150,14 +166,7 @@ static jmethodID getCachedMethodID(JNIEnv *env, if (it != cachedMethodID->constEnd()) return it.value(); - jmethodID id = 0; - if (isStatic) - id = env->GetStaticMethodID(clazz, name, sig); - else - id = env->GetMethodID(clazz, name, sig); - - if (exceptionCheckAndClear(env)) - id = 0; + jmethodID id = getMethodID(env, clazz, name, sig, isStatic); cachedMethodID->insert(key, id); return id; @@ -168,13 +177,32 @@ typedef QHash JFieldIDHash; Q_GLOBAL_STATIC(JFieldIDHash, cachedFieldID) Q_GLOBAL_STATIC(QReadWriteLock, cachedFieldIDLock) +static inline jfieldID getFieldID(JNIEnv *env, + jclass clazz, + const char *name, + const char *sig, + bool isStatic = false) +{ + jfieldID id = isStatic ? env->GetStaticFieldID(clazz, name, sig) + : env->GetFieldID(clazz, name, sig); + + if (exceptionCheckAndClear(env)) + return 0; + + return id; +} + static jfieldID getCachedFieldID(JNIEnv *env, jclass clazz, + const QByteArray &className, const char *name, const char *sig, bool isStatic = false) { - const QString key = keyBase().arg(size_t(clazz)).arg(QLatin1String(name)).arg(QLatin1String(sig)); + if (className.isNull()) + return getFieldID(env, clazz, name, sig, isStatic); + + const QString key = keyBase().arg(QLatin1String(className)).arg(QLatin1String(name)).arg(QLatin1String(sig)); QHash::const_iterator it; { @@ -190,14 +218,7 @@ static jfieldID getCachedFieldID(JNIEnv *env, if (it != cachedFieldID->constEnd()) return it.value(); - jfieldID id = 0; - if (isStatic) - id = env->GetStaticFieldID(clazz, name, sig); - else - id = env->GetFieldID(clazz, name, sig); - - if (exceptionCheckAndClear(env)) - id = 0; + jfieldID id = getFieldID(env, clazz, name, sig, isStatic); cachedFieldID->insert(key, id); return id; @@ -241,7 +262,7 @@ JNIEnv *QJNIEnvironmentPrivate::operator->() jclass QJNIEnvironmentPrivate::findClass(const char *className, JNIEnv *env) { - const QString &classDotEnc = toDotEncodedClassName(className); + const QByteArray &classDotEnc = toBinaryEncClassName(className); bool isCached = false; jclass clazz = getCachedClass(classDotEnc, &isCached); @@ -250,9 +271,10 @@ jclass QJNIEnvironmentPrivate::findClass(const char *className, JNIEnv *env) if (found) return clazz; + const QLatin1String key(classDotEnc); if (env != 0) { // We got an env. pointer (We expect this to be the right env. and call FindClass()) QWriteLocker locker(cachedClassesLock); - const QHash::const_iterator &it = cachedClasses->constFind(classDotEnc); + const QHash::const_iterator &it = cachedClasses->constFind(key); // Did we lose the race? if (it != cachedClasses->constEnd()) return it.value(); @@ -264,11 +286,11 @@ jclass QJNIEnvironmentPrivate::findClass(const char *className, JNIEnv *env) } if (clazz != 0) - cachedClasses->insert(classDotEnc, clazz); + cachedClasses->insert(key, clazz); } if (clazz == 0) // We didn't get an env. pointer or we got one with the WRONG class loader... - clazz = loadClassDotEnc(classDotEnc, QJNIEnvironmentPrivate()); + clazz = loadClass(classDotEnc, QJNIEnvironmentPrivate(), true); return clazz; } @@ -309,11 +331,12 @@ QJNIObjectPrivate::QJNIObjectPrivate(const char *className) : d(new QJNIObjectData()) { QJNIEnvironmentPrivate env; - d->m_jclass = loadClass(className, env); + d->m_className = toBinaryEncClassName(className); + d->m_jclass = loadClass(d->m_className, env, true); d->m_own_jclass = false; if (d->m_jclass) { // get default constructor - jmethodID constructorId = getCachedMethodID(env, d->m_jclass, "", "()V"); + jmethodID constructorId = getCachedMethodID(env, d->m_jclass, d->m_className, "", "()V"); if (constructorId) { jobject obj = env->NewObject(d->m_jclass, constructorId); if (obj) { @@ -328,10 +351,11 @@ QJNIObjectPrivate::QJNIObjectPrivate(const char *className, const char *sig, ... : d(new QJNIObjectData()) { QJNIEnvironmentPrivate env; - d->m_jclass = loadClass(className, env); + d->m_className = toBinaryEncClassName(className); + d->m_jclass = loadClass(d->m_className, env, true); d->m_own_jclass = false; if (d->m_jclass) { - jmethodID constructorId = getCachedMethodID(env, d->m_jclass, "", sig); + jmethodID constructorId = getCachedMethodID(env, d->m_jclass, d->m_className, "", sig); if (constructorId) { va_list args; va_start(args, sig); @@ -349,10 +373,11 @@ QJNIObjectPrivate::QJNIObjectPrivate(const char *className, const char *sig, con : d(new QJNIObjectData()) { QJNIEnvironmentPrivate env; - d->m_jclass = loadClass(className, env); + d->m_className = toBinaryEncClassName(className); + d->m_jclass = loadClass(d->m_className, env, true); d->m_own_jclass = false; if (d->m_jclass) { - jmethodID constructorId = getCachedMethodID(env, d->m_jclass, "", sig); + jmethodID constructorId = getCachedMethodID(env, d->m_jclass, d->m_className, "", sig); if (constructorId) { jobject obj = env->NewObjectV(d->m_jclass, constructorId, args); if (obj) { @@ -370,7 +395,7 @@ QJNIObjectPrivate::QJNIObjectPrivate(jclass clazz) d->m_jclass = static_cast(env->NewGlobalRef(clazz)); if (d->m_jclass) { // get default constructor - jmethodID constructorId = getCachedMethodID(env, d->m_jclass, "", "()V"); + jmethodID constructorId = getMethodID(env, d->m_jclass, "", "()V"); if (constructorId) { jobject obj = env->NewObject(d->m_jclass, constructorId); if (obj) { @@ -388,7 +413,7 @@ QJNIObjectPrivate::QJNIObjectPrivate(jclass clazz, const char *sig, ...) if (clazz) { d->m_jclass = static_cast(env->NewGlobalRef(clazz)); if (d->m_jclass) { - jmethodID constructorId = getCachedMethodID(env, d->m_jclass, "", sig); + jmethodID constructorId = getMethodID(env, d->m_jclass, "", sig); if (constructorId) { va_list args; va_start(args, sig); @@ -410,7 +435,7 @@ QJNIObjectPrivate::QJNIObjectPrivate(jclass clazz, const char *sig, const QVaLis if (clazz) { d->m_jclass = static_cast(env->NewGlobalRef(clazz)); if (d->m_jclass) { - jmethodID constructorId = getCachedMethodID(env, d->m_jclass, "", sig); + jmethodID constructorId = getMethodID(env, d->m_jclass, "", sig); if (constructorId) { jobject obj = env->NewObjectV(d->m_jclass, constructorId, args); if (obj) { @@ -430,15 +455,15 @@ QJNIObjectPrivate::QJNIObjectPrivate(jobject obj) QJNIEnvironmentPrivate env; d->m_jobject = env->NewGlobalRef(obj); - jclass objectClass = env->GetObjectClass(d->m_jobject); - d->m_jclass = static_cast(env->NewGlobalRef(objectClass)); - env->DeleteLocalRef(objectClass); + jclass cls = env->GetObjectClass(obj); + d->m_jclass = static_cast(env->NewGlobalRef(cls)); + env->DeleteLocalRef(cls); } template <> void QJNIObjectPrivate::callMethodV(const char *methodName, const char *sig, va_list args) const { QJNIEnvironmentPrivate env; - jmethodID id = getCachedMethodID(env, d->m_jclass, methodName, sig); + jmethodID id = getCachedMethodID(env, d->m_jclass, d->m_className, methodName, sig); if (id) { env->CallVoidMethodV(d->m_jobject, id, args); } @@ -458,7 +483,7 @@ jboolean QJNIObjectPrivate::callMethodV(const char *methodName, const { QJNIEnvironmentPrivate env; jboolean res = 0; - jmethodID id = getCachedMethodID(env, d->m_jclass, methodName, sig); + jmethodID id = getCachedMethodID(env, d->m_jclass, d->m_className, methodName, sig); if (id) { res = env->CallBooleanMethodV(d->m_jobject, id, args); } @@ -480,7 +505,7 @@ jbyte QJNIObjectPrivate::callMethodV(const char *methodName, const char * { QJNIEnvironmentPrivate env; jbyte res = 0; - jmethodID id = getCachedMethodID(env, d->m_jclass, methodName, sig); + jmethodID id = getCachedMethodID(env, d->m_jclass, d->m_className, methodName, sig); if (id) { res = env->CallByteMethodV(d->m_jobject, id, args); } @@ -502,7 +527,7 @@ jchar QJNIObjectPrivate::callMethodV(const char *methodName, const char * { QJNIEnvironmentPrivate env; jchar res = 0; - jmethodID id = getCachedMethodID(env, d->m_jclass, methodName, sig); + jmethodID id = getCachedMethodID(env, d->m_jclass, d->m_className, methodName, sig); if (id) { res = env->CallCharMethodV(d->m_jobject, id, args); } @@ -524,7 +549,7 @@ jshort QJNIObjectPrivate::callMethodV(const char *methodName, const char { QJNIEnvironmentPrivate env; jshort res = 0; - jmethodID id = getCachedMethodID(env, d->m_jclass, methodName, sig); + jmethodID id = getCachedMethodID(env, d->m_jclass, d->m_className, methodName, sig); if (id) { res = env->CallShortMethodV(d->m_jobject, id, args); } @@ -546,7 +571,7 @@ jint QJNIObjectPrivate::callMethodV(const char *methodName, const char *si { QJNIEnvironmentPrivate env; jint res = 0; - jmethodID id = getCachedMethodID(env, d->m_jclass, methodName, sig); + jmethodID id = getCachedMethodID(env, d->m_jclass, d->m_className, methodName, sig); if (id) { res = env->CallIntMethodV(d->m_jobject, id, args); } @@ -568,7 +593,7 @@ jlong QJNIObjectPrivate::callMethodV(const char *methodName, const char * { QJNIEnvironmentPrivate env; jlong res = 0; - jmethodID id = getCachedMethodID(env, d->m_jclass, methodName, sig); + jmethodID id = getCachedMethodID(env, d->m_jclass, d->m_className, methodName, sig); if (id) { res = env->CallLongMethodV(d->m_jobject, id, args); } @@ -590,7 +615,7 @@ jfloat QJNIObjectPrivate::callMethodV(const char *methodName, const char { QJNIEnvironmentPrivate env; jfloat res = 0.f; - jmethodID id = getCachedMethodID(env, d->m_jclass, methodName, sig); + jmethodID id = getCachedMethodID(env, d->m_jclass, d->m_className, methodName, sig); if (id) { res = env->CallFloatMethodV(d->m_jobject, id, args); } @@ -612,7 +637,7 @@ jdouble QJNIObjectPrivate::callMethodV(const char *methodName, const ch { QJNIEnvironmentPrivate env; jdouble res = 0.; - jmethodID id = getCachedMethodID(env, d->m_jclass, methodName, sig); + jmethodID id = getCachedMethodID(env, d->m_jclass, d->m_className, methodName, sig); if (id) { res = env->CallDoubleMethodV(d->m_jobject, id, args); } @@ -692,7 +717,7 @@ void QJNIObjectPrivate::callStaticMethodV(const char *className, QJNIEnvironmentPrivate env; jclass clazz = loadClass(className, env); if (clazz) { - jmethodID id = getCachedMethodID(env, clazz, methodName, sig, true); + jmethodID id = getCachedMethodID(env, clazz, toBinaryEncClassName(className), methodName, sig, true); if (id) { env->CallStaticVoidMethodV(clazz, id, args); } @@ -718,7 +743,7 @@ void QJNIObjectPrivate::callStaticMethodV(jclass clazz, va_list args) { QJNIEnvironmentPrivate env; - jmethodID id = getCachedMethodID(env, clazz, methodName, sig, true); + jmethodID id = getMethodID(env, clazz, methodName, sig, true); if (id) { env->CallStaticVoidMethodV(clazz, id, args); } @@ -746,7 +771,7 @@ jboolean QJNIObjectPrivate::callStaticMethodV(const char *className, jboolean res = 0; jclass clazz = loadClass(className, env); if (clazz) { - jmethodID id = getCachedMethodID(env, clazz, methodName, sig, true); + jmethodID id = getCachedMethodID(env, clazz, toBinaryEncClassName(className), methodName, sig, true); if (id) { res = env->CallStaticBooleanMethodV(clazz, id, args); } @@ -776,7 +801,7 @@ jboolean QJNIObjectPrivate::callStaticMethodV(jclass clazz, { QJNIEnvironmentPrivate env; jboolean res = 0; - jmethodID id = getCachedMethodID(env, clazz, methodName, sig, true); + jmethodID id = getMethodID(env, clazz, methodName, sig, true); if (id) { res = env->CallStaticBooleanMethodV(clazz, id, args); } @@ -807,7 +832,7 @@ jbyte QJNIObjectPrivate::callStaticMethodV(const char *className, jbyte res = 0; jclass clazz = loadClass(className, env); if (clazz) { - jmethodID id = getCachedMethodID(env, clazz, methodName, sig, true); + jmethodID id = getCachedMethodID(env, clazz, toBinaryEncClassName(className), methodName, sig, true); if (id) { res = env->CallStaticByteMethodV(clazz, id, args); } @@ -837,7 +862,7 @@ jbyte QJNIObjectPrivate::callStaticMethodV(jclass clazz, { QJNIEnvironmentPrivate env; jbyte res = 0; - jmethodID id = getCachedMethodID(env, clazz, methodName, sig, true); + jmethodID id = getMethodID(env, clazz, methodName, sig, true); if (id) { res = env->CallStaticByteMethodV(clazz, id, args); } @@ -868,7 +893,7 @@ jchar QJNIObjectPrivate::callStaticMethodV(const char *className, jchar res = 0; jclass clazz = loadClass(className, env); if (clazz) { - jmethodID id = getCachedMethodID(env, clazz, methodName, sig, true); + jmethodID id = getCachedMethodID(env, clazz, toBinaryEncClassName(className), methodName, sig, true); if (id) { res = env->CallStaticCharMethodV(clazz, id, args); } @@ -898,7 +923,7 @@ jchar QJNIObjectPrivate::callStaticMethodV(jclass clazz, { QJNIEnvironmentPrivate env; jchar res = 0; - jmethodID id = getCachedMethodID(env, clazz, methodName, sig, true); + jmethodID id = getMethodID(env, clazz, methodName, sig, true); if (id) { res = env->CallStaticCharMethodV(clazz, id, args); } @@ -929,7 +954,7 @@ jshort QJNIObjectPrivate::callStaticMethodV(const char *className, jshort res = 0; jclass clazz = loadClass(className, env); if (clazz) { - jmethodID id = getCachedMethodID(env, clazz, methodName, sig, true); + jmethodID id = getCachedMethodID(env, clazz, toBinaryEncClassName(className), methodName, sig, true); if (id) { res = env->CallStaticShortMethodV(clazz, id, args); } @@ -959,7 +984,7 @@ jshort QJNIObjectPrivate::callStaticMethodV(jclass clazz, { QJNIEnvironmentPrivate env; jshort res = 0; - jmethodID id = getCachedMethodID(env, clazz, methodName, sig, true); + jmethodID id = getMethodID(env, clazz, methodName, sig, true); if (id) { res = env->CallStaticShortMethodV(clazz, id, args); } @@ -990,7 +1015,7 @@ jint QJNIObjectPrivate::callStaticMethodV(const char *className, jint res = 0; jclass clazz = loadClass(className, env); if (clazz) { - jmethodID id = getCachedMethodID(env, clazz, methodName, sig, true); + jmethodID id = getCachedMethodID(env, clazz, toBinaryEncClassName(className), methodName, sig, true); if (id) { res = env->CallStaticIntMethodV(clazz, id, args); } @@ -1020,7 +1045,7 @@ jint QJNIObjectPrivate::callStaticMethodV(jclass clazz, { QJNIEnvironmentPrivate env; jint res = 0; - jmethodID id = getCachedMethodID(env, clazz, methodName, sig, true); + jmethodID id = getMethodID(env, clazz, methodName, sig, true); if (id) { res = env->CallStaticIntMethodV(clazz, id, args); } @@ -1051,7 +1076,7 @@ jlong QJNIObjectPrivate::callStaticMethodV(const char *className, jlong res = 0; jclass clazz = loadClass(className, env); if (clazz) { - jmethodID id = getCachedMethodID(env, clazz, methodName, sig, true); + jmethodID id = getCachedMethodID(env, clazz, toBinaryEncClassName(className), methodName, sig, true); if (id) { res = env->CallStaticLongMethodV(clazz, id, args); } @@ -1081,7 +1106,7 @@ jlong QJNIObjectPrivate::callStaticMethodV(jclass clazz, { QJNIEnvironmentPrivate env; jlong res = 0; - jmethodID id = getCachedMethodID(env, clazz, methodName, sig, true); + jmethodID id = getMethodID(env, clazz, methodName, sig, true); if (id) { res = env->CallStaticLongMethodV(clazz, id, args); } @@ -1112,7 +1137,7 @@ jfloat QJNIObjectPrivate::callStaticMethodV(const char *className, jfloat res = 0.f; jclass clazz = loadClass(className, env); if (clazz) { - jmethodID id = getCachedMethodID(env, clazz, methodName, sig, true); + jmethodID id = getCachedMethodID(env, clazz, toBinaryEncClassName(className), methodName, sig, true); if (id) { res = env->CallStaticFloatMethodV(clazz, id, args); } @@ -1142,7 +1167,7 @@ jfloat QJNIObjectPrivate::callStaticMethodV(jclass clazz, { QJNIEnvironmentPrivate env; jfloat res = 0.f; - jmethodID id = getCachedMethodID(env, clazz, methodName, sig, true); + jmethodID id = getMethodID(env, clazz, methodName, sig, true); if (id) { res = env->CallStaticFloatMethodV(clazz, id, args); } @@ -1173,7 +1198,7 @@ jdouble QJNIObjectPrivate::callStaticMethodV(const char *className, jdouble res = 0.; jclass clazz = loadClass(className, env); if (clazz) { - jmethodID id = getCachedMethodID(env, clazz, methodName, sig, true); + jmethodID id = getCachedMethodID(env, clazz, toBinaryEncClassName(className), methodName, sig, true); if (id) { res = env->CallStaticDoubleMethodV(clazz, id, args); } @@ -1203,7 +1228,7 @@ jdouble QJNIObjectPrivate::callStaticMethodV(jclass clazz, { QJNIEnvironmentPrivate env; jdouble res = 0.; - jmethodID id = getCachedMethodID(env, clazz, methodName, sig, true); + jmethodID id = getMethodID(env, clazz, methodName, sig, true); if (id) { res = env->CallStaticDoubleMethodV(clazz, id, args); } @@ -1338,7 +1363,7 @@ QJNIObjectPrivate QJNIObjectPrivate::callObjectMethodV(const char *methodName, { QJNIEnvironmentPrivate env; jobject res = 0; - jmethodID id = getCachedMethodID(env, d->m_jclass, methodName, sig); + jmethodID id = getCachedMethodID(env, d->m_jclass, d->m_className, methodName, sig); if (id) { res = env->CallObjectMethodV(d->m_jobject, id, args); if (res && env->ExceptionCheck()) @@ -1418,7 +1443,7 @@ QJNIObjectPrivate QJNIObjectPrivate::callStaticObjectMethodV(const char *classNa jobject res = 0; jclass clazz = loadClass(className, env); if (clazz) { - jmethodID id = getCachedMethodID(env, clazz, methodName, sig, true); + jmethodID id = getCachedMethodID(env, clazz, toBinaryEncClassName(className), methodName, sig, true); if (id) { res = env->CallStaticObjectMethodV(clazz, id, args); if (res && env->ExceptionCheck()) @@ -1450,7 +1475,7 @@ QJNIObjectPrivate QJNIObjectPrivate::callStaticObjectMethodV(jclass clazz, { QJNIEnvironmentPrivate env; jobject res = 0; - jmethodID id = getCachedMethodID(env, clazz, methodName, sig, true); + jmethodID id = getMethodID(env, clazz, methodName, sig, true); if (id) { res = env->CallStaticObjectMethodV(clazz, id, args); if (res && env->ExceptionCheck()) @@ -1479,7 +1504,7 @@ jboolean QJNIObjectPrivate::getField(const char *fieldName) const { QJNIEnvironmentPrivate env; jboolean res = 0; - jfieldID id = getCachedFieldID(env, d->m_jclass, fieldName, "Z"); + jfieldID id = getCachedFieldID(env, d->m_jclass, d->m_className, fieldName, "Z"); if (id) res = env->GetBooleanField(d->m_jobject, id); @@ -1491,7 +1516,7 @@ jbyte QJNIObjectPrivate::getField(const char *fieldName) const { QJNIEnvironmentPrivate env; jbyte res = 0; - jfieldID id = getCachedFieldID(env, d->m_jclass, fieldName, "B"); + jfieldID id = getCachedFieldID(env, d->m_jclass, d->m_className, fieldName, "B"); if (id) res = env->GetByteField(d->m_jobject, id); @@ -1503,7 +1528,7 @@ jchar QJNIObjectPrivate::getField(const char *fieldName) const { QJNIEnvironmentPrivate env; jchar res = 0; - jfieldID id = getCachedFieldID(env, d->m_jclass, fieldName, "C"); + jfieldID id = getCachedFieldID(env, d->m_jclass, d->m_className, fieldName, "C"); if (id) res = env->GetCharField(d->m_jobject, id); @@ -1515,7 +1540,7 @@ jshort QJNIObjectPrivate::getField(const char *fieldName) const { QJNIEnvironmentPrivate env; jshort res = 0; - jfieldID id = getCachedFieldID(env, d->m_jclass, fieldName, "S"); + jfieldID id = getCachedFieldID(env, d->m_jclass, d->m_className, fieldName, "S"); if (id) res = env->GetShortField(d->m_jobject, id); @@ -1527,7 +1552,7 @@ jint QJNIObjectPrivate::getField(const char *fieldName) const { QJNIEnvironmentPrivate env; jint res = 0; - jfieldID id = getCachedFieldID(env, d->m_jclass, fieldName, "I"); + jfieldID id = getCachedFieldID(env, d->m_jclass, d->m_className, fieldName, "I"); if (id) res = env->GetIntField(d->m_jobject, id); @@ -1539,7 +1564,7 @@ jlong QJNIObjectPrivate::getField(const char *fieldName) const { QJNIEnvironmentPrivate env; jlong res = 0; - jfieldID id = getCachedFieldID(env, d->m_jclass, fieldName, "J"); + jfieldID id = getCachedFieldID(env, d->m_jclass, d->m_className, fieldName, "J"); if (id) res = env->GetLongField(d->m_jobject, id); @@ -1551,7 +1576,7 @@ jfloat QJNIObjectPrivate::getField(const char *fieldName) const { QJNIEnvironmentPrivate env; jfloat res = 0.f; - jfieldID id = getCachedFieldID(env, d->m_jclass, fieldName, "F"); + jfieldID id = getCachedFieldID(env, d->m_jclass, d->m_className, fieldName, "F"); if (id) res = env->GetFloatField(d->m_jobject, id); @@ -1563,7 +1588,7 @@ jdouble QJNIObjectPrivate::getField(const char *fieldName) const { QJNIEnvironmentPrivate env; jdouble res = 0.; - jfieldID id = getCachedFieldID(env, d->m_jclass, fieldName, "D"); + jfieldID id = getCachedFieldID(env, d->m_jclass, d->m_className, fieldName, "D"); if (id) res = env->GetDoubleField(d->m_jobject, id); @@ -1575,7 +1600,7 @@ jboolean QJNIObjectPrivate::getStaticField(jclass clazz, const char *f { QJNIEnvironmentPrivate env; jboolean res = 0; - jfieldID id = getCachedFieldID(env, clazz, fieldName, "Z", true); + jfieldID id = getFieldID(env, clazz, fieldName, "Z", true); if (id) res = env->GetStaticBooleanField(clazz, id); @@ -1586,12 +1611,15 @@ template <> jboolean QJNIObjectPrivate::getStaticField(const char *className, const char *fieldName) { QJNIEnvironmentPrivate env; - jboolean res = 0; jclass clazz = loadClass(className, env); - if (clazz) - res = getStaticField(clazz, fieldName); + if (clazz == 0) + return 0; - return res; + jfieldID id = getCachedFieldID(env, clazz, toBinaryEncClassName(className), fieldName, "Z", true); + if (id == 0) + return 0; + + return env->GetStaticBooleanField(clazz, id); } template <> @@ -1599,7 +1627,7 @@ jbyte QJNIObjectPrivate::getStaticField(jclass clazz, const char *fieldNa { QJNIEnvironmentPrivate env; jbyte res = 0; - jfieldID id = getCachedFieldID(env, clazz, fieldName, "B", true); + jfieldID id = getFieldID(env, clazz, fieldName, "B", true); if (id) res = env->GetStaticByteField(clazz, id); @@ -1610,12 +1638,15 @@ template <> jbyte QJNIObjectPrivate::getStaticField(const char *className, const char *fieldName) { QJNIEnvironmentPrivate env; - jbyte res = 0; jclass clazz = loadClass(className, env); - if (clazz) - res = getStaticField(clazz, fieldName); + if (clazz == 0) + return 0; - return res; + jfieldID id = getCachedFieldID(env, clazz, toBinaryEncClassName(className), fieldName, "B", true); + if (id == 0) + return 0; + + return env->GetStaticByteField(clazz, id); } template <> @@ -1623,7 +1654,7 @@ jchar QJNIObjectPrivate::getStaticField(jclass clazz, const char *fieldNa { QJNIEnvironmentPrivate env; jchar res = 0; - jfieldID id = getCachedFieldID(env, clazz, fieldName, "C", true); + jfieldID id = getFieldID(env, clazz, fieldName, "C", true); if (id) res = env->GetStaticCharField(clazz, id); @@ -1634,12 +1665,15 @@ template <> jchar QJNIObjectPrivate::getStaticField(const char *className, const char *fieldName) { QJNIEnvironmentPrivate env; - jchar res = 0; jclass clazz = loadClass(className, env); - if (clazz) - res = getStaticField(clazz, fieldName); + if (clazz == 0) + return 0; - return res; + jfieldID id = getCachedFieldID(env, clazz, toBinaryEncClassName(className), fieldName, "C", true); + if (id == 0) + return 0; + + return env->GetStaticCharField(clazz, id); } template <> @@ -1647,7 +1681,7 @@ jshort QJNIObjectPrivate::getStaticField(jclass clazz, const char *field { QJNIEnvironmentPrivate env; jshort res = 0; - jfieldID id = getCachedFieldID(env, clazz, fieldName, "S", true); + jfieldID id = getFieldID(env, clazz, fieldName, "S", true); if (id) res = env->GetStaticShortField(clazz, id); @@ -1658,12 +1692,15 @@ template <> jshort QJNIObjectPrivate::getStaticField(const char *className, const char *fieldName) { QJNIEnvironmentPrivate env; - jshort res = 0; jclass clazz = loadClass(className, env); - if (clazz) - res = getStaticField(clazz, fieldName); + if (clazz == 0) + return 0; - return res; + jfieldID id = getCachedFieldID(env, clazz, toBinaryEncClassName(className), fieldName, "S", true); + if (id == 0) + return 0; + + return env->GetStaticShortField(clazz, id); } template <> @@ -1671,7 +1708,7 @@ jint QJNIObjectPrivate::getStaticField(jclass clazz, const char *fieldName { QJNIEnvironmentPrivate env; jint res = 0; - jfieldID id = getCachedFieldID(env, clazz, fieldName, "I", true); + jfieldID id = getFieldID(env, clazz, fieldName, "I", true); if (id) res = env->GetStaticIntField(clazz, id); @@ -1682,12 +1719,15 @@ template <> jint QJNIObjectPrivate::getStaticField(const char *className, const char *fieldName) { QJNIEnvironmentPrivate env; - jint res = 0; jclass clazz = loadClass(className, env); - if (clazz) - res = getStaticField(clazz, fieldName); + if (clazz == 0) + return 0; - return res; + jfieldID id = getCachedFieldID(env, clazz, toBinaryEncClassName(className), fieldName, "I", true); + if (id == 0) + return 0; + + return env->GetStaticIntField(clazz, id); } template <> @@ -1695,7 +1735,7 @@ jlong QJNIObjectPrivate::getStaticField(jclass clazz, const char *fieldNa { QJNIEnvironmentPrivate env; jlong res = 0; - jfieldID id = getCachedFieldID(env, clazz, fieldName, "J", true); + jfieldID id = getFieldID(env, clazz, fieldName, "J", true); if (id) res = env->GetStaticLongField(clazz, id); @@ -1706,12 +1746,15 @@ template <> jlong QJNIObjectPrivate::getStaticField(const char *className, const char *fieldName) { QJNIEnvironmentPrivate env; - jlong res = 0; jclass clazz = loadClass(className, env); - if (clazz) - res = getStaticField(clazz, fieldName); + if (clazz == 0) + return 0; - return res; + jfieldID id = getCachedFieldID(env, clazz, toBinaryEncClassName(className), fieldName, "J", true); + if (id == 0) + return 0; + + return env->GetStaticLongField(clazz, id); } template <> @@ -1719,7 +1762,7 @@ jfloat QJNIObjectPrivate::getStaticField(jclass clazz, const char *field { QJNIEnvironmentPrivate env; jfloat res = 0.f; - jfieldID id = getCachedFieldID(env, clazz, fieldName, "F", true); + jfieldID id = getFieldID(env, clazz, fieldName, "F", true); if (id) res = env->GetStaticFloatField(clazz, id); @@ -1730,12 +1773,15 @@ template <> jfloat QJNIObjectPrivate::getStaticField(const char *className, const char *fieldName) { QJNIEnvironmentPrivate env; - jfloat res = 0.f; jclass clazz = loadClass(className, env); - if (clazz) - res = getStaticField(clazz, fieldName); + if (clazz == 0) + return 0.f; - return res; + jfieldID id = getCachedFieldID(env, clazz, toBinaryEncClassName(className), fieldName, "F", true); + if (id == 0) + return 0.f; + + return env->GetStaticFloatField(clazz, id); } template <> @@ -1743,7 +1789,7 @@ jdouble QJNIObjectPrivate::getStaticField(jclass clazz, const char *fie { QJNIEnvironmentPrivate env; jdouble res = 0.; - jfieldID id = getCachedFieldID(env, clazz, fieldName, "D", true); + jfieldID id = getFieldID(env, clazz, fieldName, "D", true); if (id) res = env->GetStaticDoubleField(clazz, id); @@ -1754,12 +1800,15 @@ template <> jdouble QJNIObjectPrivate::getStaticField(const char *className, const char *fieldName) { QJNIEnvironmentPrivate env; - jdouble res = 0.; jclass clazz = loadClass(className, env); - if (clazz) - res = getStaticField(clazz, fieldName); + if (clazz == 0) + return 0.; - return res; + jfieldID id = getCachedFieldID(env, clazz, toBinaryEncClassName(className), fieldName, "D", true); + if (id == 0) + return 0.; + + return env->GetStaticDoubleField(clazz, id); } QJNIObjectPrivate QJNIObjectPrivate::getObjectField(const char *fieldName, @@ -1767,7 +1816,7 @@ QJNIObjectPrivate QJNIObjectPrivate::getObjectField(const char *fieldName, { QJNIEnvironmentPrivate env; jobject res = 0; - jfieldID id = getCachedFieldID(env, d->m_jclass, fieldName, sig); + jfieldID id = getCachedFieldID(env, d->m_jclass, d->m_className, fieldName, sig); if (id) { res = env->GetObjectField(d->m_jobject, id); if (res && env->ExceptionCheck()) @@ -1784,12 +1833,21 @@ QJNIObjectPrivate QJNIObjectPrivate::getStaticObjectField(const char *className, const char *sig) { QJNIEnvironmentPrivate env; - QJNIObjectPrivate res; jclass clazz = loadClass(className, env); - if (clazz) - res = getStaticObjectField(clazz, fieldName, sig); + if (clazz == 0) + return QJNIObjectPrivate(); - return res; + jfieldID id = getCachedFieldID(env, clazz, toBinaryEncClassName(className), fieldName, sig, true); + if (id == 0) + return QJNIObjectPrivate(); + + jobject res = env->GetStaticObjectField(clazz, id); + if (res && env->ExceptionCheck()) + res = 0; + + QJNIObjectPrivate obj(res); + env->DeleteLocalRef(res); + return obj; } QJNIObjectPrivate QJNIObjectPrivate::getStaticObjectField(jclass clazz, @@ -1798,7 +1856,7 @@ QJNIObjectPrivate QJNIObjectPrivate::getStaticObjectField(jclass clazz, { QJNIEnvironmentPrivate env; jobject res = 0; - jfieldID id = getCachedFieldID(env, clazz, fieldName, sig, true); + jfieldID id = getFieldID(env, clazz, fieldName, sig, true); if (id) { res = env->GetStaticObjectField(clazz, id); if (res && env->ExceptionCheck()) @@ -1814,7 +1872,7 @@ template <> void QJNIObjectPrivate::setField(const char *fieldName, jboolean value) { QJNIEnvironmentPrivate env; - jfieldID id = getCachedFieldID(env, d->m_jclass, fieldName, "Z"); + jfieldID id = getCachedFieldID(env, d->m_jclass, d->m_className, fieldName, "Z"); if (id) env->SetBooleanField(d->m_jobject, id, value); @@ -1824,7 +1882,7 @@ template <> void QJNIObjectPrivate::setField(const char *fieldName, jbyte value) { QJNIEnvironmentPrivate env; - jfieldID id = getCachedFieldID(env, d->m_jclass, fieldName, "B"); + jfieldID id = getCachedFieldID(env, d->m_jclass, d->m_className, fieldName, "B"); if (id) env->SetByteField(d->m_jobject, id, value); @@ -1834,7 +1892,7 @@ template <> void QJNIObjectPrivate::setField(const char *fieldName, jchar value) { QJNIEnvironmentPrivate env; - jfieldID id = getCachedFieldID(env, d->m_jclass, fieldName, "C"); + jfieldID id = getCachedFieldID(env, d->m_jclass, d->m_className, fieldName, "C"); if (id) env->SetCharField(d->m_jobject, id, value); @@ -1844,7 +1902,7 @@ template <> void QJNIObjectPrivate::setField(const char *fieldName, jshort value) { QJNIEnvironmentPrivate env; - jfieldID id = getCachedFieldID(env, d->m_jclass, fieldName, "S"); + jfieldID id = getCachedFieldID(env, d->m_jclass, d->m_className, fieldName, "S"); if (id) env->SetShortField(d->m_jobject, id, value); @@ -1854,7 +1912,7 @@ template <> void QJNIObjectPrivate::setField(const char *fieldName, jint value) { QJNIEnvironmentPrivate env; - jfieldID id = getCachedFieldID(env, d->m_jclass, fieldName, "I"); + jfieldID id = getCachedFieldID(env, d->m_jclass, d->m_className, fieldName, "I"); if (id) env->SetIntField(d->m_jobject, id, value); @@ -1864,7 +1922,7 @@ template <> void QJNIObjectPrivate::setField(const char *fieldName, jlong value) { QJNIEnvironmentPrivate env; - jfieldID id = getCachedFieldID(env, d->m_jclass, fieldName, "J"); + jfieldID id = getCachedFieldID(env, d->m_jclass, d->m_className, fieldName, "J"); if (id) env->SetLongField(d->m_jobject, id, value); @@ -1874,7 +1932,7 @@ template <> void QJNIObjectPrivate::setField(const char *fieldName, jfloat value) { QJNIEnvironmentPrivate env; - jfieldID id = getCachedFieldID(env, d->m_jclass, fieldName, "F"); + jfieldID id = getCachedFieldID(env, d->m_jclass, d->m_className, fieldName, "F"); if (id) env->SetFloatField(d->m_jobject, id, value); @@ -1884,7 +1942,7 @@ template <> void QJNIObjectPrivate::setField(const char *fieldName, jdouble value) { QJNIEnvironmentPrivate env; - jfieldID id = getCachedFieldID(env, d->m_jclass, fieldName, "D"); + jfieldID id = getCachedFieldID(env, d->m_jclass, d->m_className, fieldName, "D"); if (id) env->SetDoubleField(d->m_jobject, id, value); @@ -1894,7 +1952,7 @@ template <> void QJNIObjectPrivate::setField(const char *fieldName, jbooleanArray value) { QJNIEnvironmentPrivate env; - jfieldID id = getCachedFieldID(env, d->m_jclass, fieldName, "[Z"); + jfieldID id = getCachedFieldID(env, d->m_jclass, d->m_className, fieldName, "[Z"); if (id) env->SetObjectField(d->m_jobject, id, value); @@ -1904,7 +1962,7 @@ template <> void QJNIObjectPrivate::setField(const char *fieldName, jbyteArray value) { QJNIEnvironmentPrivate env; - jfieldID id = getCachedFieldID(env, d->m_jclass, fieldName, "[B"); + jfieldID id = getCachedFieldID(env, d->m_jclass, d->m_className, fieldName, "[B"); if (id) env->SetObjectField(d->m_jobject, id, value); @@ -1914,7 +1972,7 @@ template <> void QJNIObjectPrivate::setField(const char *fieldName, jcharArray value) { QJNIEnvironmentPrivate env; - jfieldID id = getCachedFieldID(env, d->m_jclass, fieldName, "[C"); + jfieldID id = getCachedFieldID(env, d->m_jclass, d->m_className, fieldName, "[C"); if (id) env->SetObjectField(d->m_jobject, id, value); @@ -1924,7 +1982,7 @@ template <> void QJNIObjectPrivate::setField(const char *fieldName, jshortArray value) { QJNIEnvironmentPrivate env; - jfieldID id = getCachedFieldID(env, d->m_jclass, fieldName, "[S"); + jfieldID id = getCachedFieldID(env, d->m_jclass, d->m_className, fieldName, "[S"); if (id) env->SetObjectField(d->m_jobject, id, value); @@ -1934,7 +1992,7 @@ template <> void QJNIObjectPrivate::setField(const char *fieldName, jintArray value) { QJNIEnvironmentPrivate env; - jfieldID id = getCachedFieldID(env, d->m_jclass, fieldName, "[I"); + jfieldID id = getCachedFieldID(env, d->m_jclass, d->m_className, fieldName, "[I"); if (id) env->SetObjectField(d->m_jobject, id, value); @@ -1944,7 +2002,7 @@ template <> void QJNIObjectPrivate::setField(const char *fieldName, jlongArray value) { QJNIEnvironmentPrivate env; - jfieldID id = getCachedFieldID(env, d->m_jclass, fieldName, "[J"); + jfieldID id = getCachedFieldID(env, d->m_jclass, d->m_className, fieldName, "[J"); if (id) env->SetObjectField(d->m_jobject, id, value); @@ -1954,7 +2012,7 @@ template <> void QJNIObjectPrivate::setField(const char *fieldName, jfloatArray value) { QJNIEnvironmentPrivate env; - jfieldID id = getCachedFieldID(env, d->m_jclass, fieldName, "[F"); + jfieldID id = getCachedFieldID(env, d->m_jclass, d->m_className, fieldName, "[F"); if (id) env->SetObjectField(d->m_jobject, id, value); @@ -1964,7 +2022,7 @@ template <> void QJNIObjectPrivate::setField(const char *fieldName, jdoubleArray value) { QJNIEnvironmentPrivate env; - jfieldID id = getCachedFieldID(env, d->m_jclass, fieldName, "[D"); + jfieldID id = getCachedFieldID(env, d->m_jclass, d->m_className, fieldName, "[D"); if (id) env->SetObjectField(d->m_jobject, id, value); @@ -1974,7 +2032,7 @@ template <> void QJNIObjectPrivate::setField(const char *fieldName, jstring value) { QJNIEnvironmentPrivate env; - jfieldID id = getCachedFieldID(env, d->m_jclass, fieldName, "Ljava/lang/String;"); + jfieldID id = getCachedFieldID(env, d->m_jclass, d->m_className, fieldName, "Ljava/lang/String;"); if (id) env->SetObjectField(d->m_jobject, id, value); @@ -1986,7 +2044,7 @@ void QJNIObjectPrivate::setField(const char *fieldName, jobject value) { QJNIEnvironmentPrivate env; - jfieldID id = getCachedFieldID(env, d->m_jclass, fieldName, sig); + jfieldID id = getCachedFieldID(env, d->m_jclass, d->m_className, fieldName, sig); if (id) env->SetObjectField(d->m_jobject, id, value); @@ -1998,7 +2056,7 @@ void QJNIObjectPrivate::setField(const char *fieldName, jobjectArray value) { QJNIEnvironmentPrivate env; - jfieldID id = getCachedFieldID(env, d->m_jclass, fieldName, sig); + jfieldID id = getCachedFieldID(env, d->m_jclass, d->m_className, fieldName, sig); if (id) env->SetObjectField(d->m_jobject, id, value); @@ -2010,7 +2068,7 @@ void QJNIObjectPrivate::setStaticField(jclass clazz, jboolean value) { QJNIEnvironmentPrivate env; - jfieldID id = getCachedFieldID(env, clazz, fieldName, "Z", true); + jfieldID id = getFieldID(env, clazz, fieldName, "Z", true); if (id) env->SetStaticBooleanField(clazz, id, value); } @@ -2022,8 +2080,14 @@ void QJNIObjectPrivate::setStaticField(const char *className, { QJNIEnvironmentPrivate env; jclass clazz = loadClass(className, env); - if (clazz) - setStaticField(clazz, fieldName, value); + if (clazz == 0) + return; + + jfieldID id = getCachedFieldID(env, clazz, className, fieldName, "Z", true); + if (id == 0) + return; + + env->SetStaticBooleanField(clazz, id, value); } template <> @@ -2032,7 +2096,7 @@ void QJNIObjectPrivate::setStaticField(jclass clazz, jbyte value) { QJNIEnvironmentPrivate env; - jfieldID id = getCachedFieldID(env, clazz, fieldName, "B", true); + jfieldID id = getFieldID(env, clazz, fieldName, "B", true); if (id) env->SetStaticByteField(clazz, id, value); } @@ -2044,8 +2108,14 @@ void QJNIObjectPrivate::setStaticField(const char *className, { QJNIEnvironmentPrivate env; jclass clazz = loadClass(className, env); - if (clazz) - setStaticField(clazz, fieldName, value); + if (clazz == 0) + return; + + jfieldID id = getCachedFieldID(env, clazz, className, fieldName, "B", true); + if (id == 0) + return; + + env->SetStaticByteField(clazz, id, value); } template <> @@ -2054,7 +2124,7 @@ void QJNIObjectPrivate::setStaticField(jclass clazz, jchar value) { QJNIEnvironmentPrivate env; - jfieldID id = getCachedFieldID(env, clazz, fieldName, "C", true); + jfieldID id = getFieldID(env, clazz, fieldName, "C", true); if (id) env->SetStaticCharField(clazz, id, value); } @@ -2066,8 +2136,14 @@ void QJNIObjectPrivate::setStaticField(const char *className, { QJNIEnvironmentPrivate env; jclass clazz = loadClass(className, env); - if (clazz) - setStaticField(clazz, fieldName, value); + if (clazz == 0) + return; + + jfieldID id = getCachedFieldID(env, clazz, className, fieldName, "C", true); + if (id == 0) + return; + + env->SetStaticCharField(clazz, id, value); } template <> @@ -2076,7 +2152,7 @@ void QJNIObjectPrivate::setStaticField(jclass clazz, jshort value) { QJNIEnvironmentPrivate env; - jfieldID id = getCachedFieldID(env, clazz, fieldName, "S", true); + jfieldID id = getFieldID(env, clazz, fieldName, "S", true); if (id) env->SetStaticShortField(clazz, id, value); } @@ -2088,8 +2164,14 @@ void QJNIObjectPrivate::setStaticField(const char *className, { QJNIEnvironmentPrivate env; jclass clazz = loadClass(className, env); - if (clazz) - setStaticField(clazz, fieldName, value); + if (clazz == 0) + return; + + jfieldID id = getCachedFieldID(env, clazz, className, fieldName, "S", true); + if (id == 0) + return; + + env->SetStaticShortField(clazz, id, value); } template <> @@ -2098,7 +2180,7 @@ void QJNIObjectPrivate::setStaticField(jclass clazz, jint value) { QJNIEnvironmentPrivate env; - jfieldID id = getCachedFieldID(env, clazz, fieldName, "I", true); + jfieldID id = getFieldID(env, clazz, fieldName, "I", true); if (id) env->SetStaticIntField(clazz, id, value); } @@ -2110,8 +2192,14 @@ void QJNIObjectPrivate::setStaticField(const char *className, { QJNIEnvironmentPrivate env; jclass clazz = loadClass(className, env); - if (clazz) - setStaticField(clazz, fieldName, value); + if (clazz == 0) + return; + + jfieldID id = getCachedFieldID(env, clazz, className, fieldName, "I", true); + if (id == 0) + return; + + env->SetStaticIntField(clazz, id, value); } template <> @@ -2120,7 +2208,7 @@ void QJNIObjectPrivate::setStaticField(jclass clazz, jlong value) { QJNIEnvironmentPrivate env; - jfieldID id = getCachedFieldID(env, clazz, fieldName, "J", true); + jfieldID id = getFieldID(env, clazz, fieldName, "J", true); if (id) env->SetStaticLongField(clazz, id, value); } @@ -2132,8 +2220,14 @@ void QJNIObjectPrivate::setStaticField(const char *className, { QJNIEnvironmentPrivate env; jclass clazz = loadClass(className, env); - if (clazz) - setStaticField(clazz, fieldName, value); + if (clazz == 0) + return; + + jfieldID id = getCachedFieldID(env, clazz, className, fieldName, "J", true); + if (id == 0) + return; + + env->SetStaticLongField(clazz, id, value); } template <> @@ -2142,7 +2236,7 @@ void QJNIObjectPrivate::setStaticField(jclass clazz, jfloat value) { QJNIEnvironmentPrivate env; - jfieldID id = getCachedFieldID(env, clazz, fieldName, "F", true); + jfieldID id = getFieldID(env, clazz, fieldName, "F", true); if (id) env->SetStaticFloatField(clazz, id, value); } @@ -2154,8 +2248,14 @@ void QJNIObjectPrivate::setStaticField(const char *className, { QJNIEnvironmentPrivate env; jclass clazz = loadClass(className, env); - if (clazz) - setStaticField(clazz, fieldName, value); + if (clazz == 0) + return; + + jfieldID id = getCachedFieldID(env, clazz, className, fieldName, "F", true); + if (id == 0) + return; + + env->SetStaticFloatField(clazz, id, value); } template <> @@ -2164,7 +2264,7 @@ void QJNIObjectPrivate::setStaticField(jclass clazz, jdouble value) { QJNIEnvironmentPrivate env; - jfieldID id = getCachedFieldID(env, clazz, fieldName, "D", true); + jfieldID id = getFieldID(env, clazz, fieldName, "D", true); if (id) env->SetStaticDoubleField(clazz, id, value); } @@ -2176,8 +2276,14 @@ void QJNIObjectPrivate::setStaticField(const char *className, { QJNIEnvironmentPrivate env; jclass clazz = loadClass(className, env); - if (clazz) - setStaticField(clazz, fieldName, value); + if (clazz == 0) + return; + + jfieldID id = getCachedFieldID(env, clazz, className, fieldName, "D", true); + if (id == 0) + return; + + env->SetStaticDoubleField(clazz, id, value); } template <> @@ -2187,7 +2293,7 @@ void QJNIObjectPrivate::setStaticField(jclass clazz, jobject value) { QJNIEnvironmentPrivate env; - jfieldID id = getCachedFieldID(env, clazz, fieldName, sig, true); + jfieldID id = getFieldID(env, clazz, fieldName, sig, true); if (id) env->SetStaticObjectField(clazz, id, value); } @@ -2200,8 +2306,14 @@ void QJNIObjectPrivate::setStaticField(const char *className, { QJNIEnvironmentPrivate env; jclass clazz = loadClass(className, env); - if (clazz) - setStaticField(clazz, fieldName, sig, value); + if (clazz == 0) + return; + + jfieldID id = getCachedFieldID(env, clazz, className, fieldName, sig, true); + if (id == 0) + return; + + env->SetStaticObjectField(clazz, id, value); } QJNIObjectPrivate QJNIObjectPrivate::fromString(const QString &string) diff --git a/src/corelib/kernel/qjni_p.h b/src/corelib/kernel/qjni_p.h index fb1982dc74..a62e9ee056 100644 --- a/src/corelib/kernel/qjni_p.h +++ b/src/corelib/kernel/qjni_p.h @@ -74,6 +74,7 @@ public: jobject m_jobject; jclass m_jclass; bool m_own_jclass; + QByteArray m_className; }; class Q_CORE_EXPORT QJNIObjectPrivate From a1ada382ff00a8e1599aba22ee06ff53ce478666 Mon Sep 17 00:00:00 2001 From: Andy Shaw Date: Thu, 30 Apr 2015 15:09:59 +0200 Subject: [PATCH 02/24] Handle parsing of GL_VERSION as reported by Nexus 6 The Nexus 6 device reports a GL_VERSION which is strictly not conformant to what is expected from GL_VERSION, so a check is added for this case so that it correctly detects the right OpenGL ES version. Change-Id: I00297dd7c1e505dd7f9ab8a7fa480f514162b488 Reviewed-by: Laszlo Agocs --- src/gui/kernel/qplatformopenglcontext.cpp | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/src/gui/kernel/qplatformopenglcontext.cpp b/src/gui/kernel/qplatformopenglcontext.cpp index 527bfdd983..364c1a5c0e 100644 --- a/src/gui/kernel/qplatformopenglcontext.cpp +++ b/src/gui/kernel/qplatformopenglcontext.cpp @@ -121,6 +121,10 @@ bool QPlatformOpenGLContext::parseOpenGLVersion(const QByteArray &versionString, if (versionParts.size() >= 2) { major = versionParts.at(0).toInt(&majorOk); minor = versionParts.at(1).toInt(&minorOk); + // Nexus 6 has "OpenGL ES 3.0V@95.0 (GIT@I86da836d38)" + if (!minorOk) + if (int idx = versionParts.at(1).indexOf('V')) + minor = versionParts.at(1).left(idx).toInt(&minorOk); } else { qWarning("Unrecognized OpenGL ES version"); } From 3eca75de67b3fd2c890715b30c7899cebc096fe9 Mon Sep 17 00:00:00 2001 From: Thiago Macieira Date: Mon, 11 May 2015 18:30:00 +0900 Subject: [PATCH 03/24] Make qglobal.h complain if you use -fPIE Prior to Qt 5.4.2 (commit 36d6eb721e7d5997ade75e289d4088dc48678d0d), we allowed it, but now we need to enforce that it is not used. Note that -fPIE does define __PIC__, so we need this to catch the use of -fPIE. [ChangeLog][Important Behavior Changes] On x86 and x86-64 systems with ELF binaries (especially Linux), due to a new optimization in GCC 5.x in combination with a recent version of GNU binutils, compiling Qt applications with -fPIE is no longer enough. Applications now need to be compiled with the -fPIC option if Qt's option "reduce relocations" is active. Note that Clang is known to generate incompatible code even with -fPIC if the -flto option is active. Task-number: QTBUG-45755 Change-Id: I66a35ce5f88941f29aa6ffff13dd210e0aa2728f Reviewed-by: Dmitry Shachnev Reviewed-by: Simon Hausmann --- dist/changes-5.4.2 | 7 +++++++ src/corelib/global/qglobal.h | 4 ++-- 2 files changed, 9 insertions(+), 2 deletions(-) diff --git a/dist/changes-5.4.2 b/dist/changes-5.4.2 index e1ad9b6846..5827187e9e 100644 --- a/dist/changes-5.4.2 +++ b/dist/changes-5.4.2 @@ -34,6 +34,13 @@ information about a particular change. common EXIF-format (big-endian) was not working until 5.4.1. 5.4.2 restores the behavior of 5.4.0 and earlier for most EXIF-tagged JPEGs. EXIF orientation will be an opt-in starting with Qt 5.5. +- On x86 and x86-64 systems with ELF binaries (especially Linux), due to + a new optimization in GCC 5.x in combination with a recent version of + GNU binutils, compiling Qt applications with -fPIE is no longer + enough. Applications now need to be compiled with the -fPIC option if + Qt's option "reduce relocations" is active. Note that Clang is known + to generate incompatible code even with -fPIC if the -flto option is + active. **************************************************************************** * Library * diff --git a/src/corelib/global/qglobal.h b/src/corelib/global/qglobal.h index ef84662036..4547877da6 100644 --- a/src/corelib/global/qglobal.h +++ b/src/corelib/global/qglobal.h @@ -1047,9 +1047,9 @@ Q_CORE_EXPORT int qrand(); # define QT_NO_SHAREDMEMORY #endif -#if !defined(QT_BOOTSTRAPPED) && defined(QT_REDUCE_RELOCATIONS) && defined(__ELF__) && !defined(__PIC__) +#if !defined(QT_BOOTSTRAPPED) && defined(QT_REDUCE_RELOCATIONS) && defined(__ELF__) && (!defined(__PIC__) || defined(__PIE__)) # error "You must build your code with position independent code if Qt was built with -reduce-relocations. "\ - "Compile your code with -fPIC." + "Compile your code with -fPIC (-fPIE is not enough)." #endif namespace QtPrivate { From 5a3251a0324d65d688c3702a327d7a71a0de5ab9 Mon Sep 17 00:00:00 2001 From: Joerg Bornemann Date: Wed, 22 Apr 2015 15:43:50 +0200 Subject: [PATCH 04/24] Doc: fix QSystemSemaphore example acquire() doesn't take arguments. Change-Id: I16f0169c40433cc3cbfcb577bd8386d217cccb12 Task-number: QTBUG-40055 Reviewed-by: Oliver Wolff Reviewed-by: Friedemann Kleint --- .../snippets/code/src_corelib_kernel_qsystemsemaphore.cpp | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/src/corelib/doc/snippets/code/src_corelib_kernel_qsystemsemaphore.cpp b/src/corelib/doc/snippets/code/src_corelib_kernel_qsystemsemaphore.cpp index bc03770f16..9cded446d1 100644 --- a/src/corelib/doc/snippets/code/src_corelib_kernel_qsystemsemaphore.cpp +++ b/src/corelib/doc/snippets/code/src_corelib_kernel_qsystemsemaphore.cpp @@ -51,8 +51,9 @@ sem.release(2); // resources available == 3 //! [1] QSystemSemaphore sem("market", 5, QSystemSemaphore::Create); -sem.acquire(5); // acquire all 5 resources -sem.release(5); // release the 5 resources +for (int i = 0; i < 5; ++i) // acquire all 5 resources + sem.acquire(); +sem.release(5); // release the 5 resources //! [1] From 2023b9e76baf8d8a3b1ea59748624e16f3297ac3 Mon Sep 17 00:00:00 2001 From: Joerg Bornemann Date: Wed, 22 Apr 2015 14:18:20 +0200 Subject: [PATCH 05/24] Doc: clarify ownership after QStackedWidget::removeWidget Change-Id: Ia14b72cdac3205a3896c47ecc81b31adb508181b Task-number: QTBUG-44891 Reviewed-by: Leena Miettinen --- src/widgets/widgets/qstackedwidget.cpp | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/src/widgets/widgets/qstackedwidget.cpp b/src/widgets/widgets/qstackedwidget.cpp index fd1bf57321..411651f480 100644 --- a/src/widgets/widgets/qstackedwidget.cpp +++ b/src/widgets/widgets/qstackedwidget.cpp @@ -182,7 +182,9 @@ int QStackedWidget::insertWidget(int index, QWidget *widget) not deleted but simply removed from the stacked layout, causing it to be hidden. - \b{Note:} Ownership of \a widget reverts to the application. + \note Parent object and parent widget of \a widget will remain the + QStackedWidget. If the application wants to reuse the removed + \a widget, then it is recommended to re-parent it. \sa addWidget(), insertWidget(), currentWidget() */ From d14397b7298879e260bc7218a5f7539b6b94a2cf Mon Sep 17 00:00:00 2001 From: Friedemann Kleint Date: Tue, 12 May 2015 17:11:00 +0200 Subject: [PATCH 06/24] Windows: Fix exit crash of GUI applications when deleting argv[]. When passing Qt arguments followed by normal arguments, a double deletion may occur due to Qt shifting argv. For example: argv[] = app -qwindowgeometry +50+50 some_arg becomes: argv[] = app some_arg some_arg Terminate deletion when encountering the null pointer. Change-Id: I5279955b6bd463f5858d6e5e8e16a1f5d0945652 Reviewed-by: Joerg Bornemann --- src/winmain/qtmain_win.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/winmain/qtmain_win.cpp b/src/winmain/qtmain_win.cpp index 25b79543ba..459ca114a6 100644 --- a/src/winmain/qtmain_win.cpp +++ b/src/winmain/qtmain_win.cpp @@ -111,7 +111,7 @@ extern "C" int APIENTRY WinMain(HINSTANCE, HINSTANCE, LPSTR /*cmdParamarg*/, int argv[argc] = Q_NULLPTR; LocalFree(argvW); const int exitCode = main(argc, argv); - for (int i = 0; i < argc; ++i) + for (int i = 0; i < argc && argv[i]; ++i) delete [] argv[i]; delete [] argv; return exitCode; From 50c41bc8efb52b2b3f2aad66d79167320e9b2b31 Mon Sep 17 00:00:00 2001 From: Nico Vertriest Date: Mon, 24 Nov 2014 14:53:05 +0100 Subject: [PATCH 07/24] Doc: corrected autolink issues statemachine.qdoc MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Task-number: QTBUG-40362 Change-Id: Ia686ebdfd722f448aa30fb1f1f266b6148df4026 Reviewed-by: Topi Reiniö --- src/corelib/doc/src/statemachine.qdoc | 32 +++++++++++++-------------- 1 file changed, 16 insertions(+), 16 deletions(-) diff --git a/src/corelib/doc/src/statemachine.qdoc b/src/corelib/doc/src/statemachine.qdoc index 846eb7d1f0..4b50174b88 100644 --- a/src/corelib/doc/src/statemachine.qdoc +++ b/src/corelib/doc/src/statemachine.qdoc @@ -63,7 +63,7 @@ used to effectively embed the elements and semantics of statecharts in Qt applications. The framework integrates tightly with Qt's meta-object system; for example, transitions between states can be triggered by signals, and - states can be configured to set properties and invoke methods on QObjects. + states can be configured to set properties and invoke methods on {QObject}s. Qt's event system is used to drive the state machines. The state graph in the State Machine framework is hierarchical. States can be nested inside of @@ -126,9 +126,9 @@ The QState::entered() signal is emitted when the state is entered, and the QState::exited() signal is emitted when the state is exited. In the - following snippet, the button's showMaximized() slot will be called when - state \c s3 is entered, and the button's showMinimized() slot will be called - when \c s3 is exited: + following snippet, the button's \l {QPushButton::}{showMaximized()} slot + will be called when state \c s3 is entered, and the button's \l {QPushButton::}{showMinimized()} + slot will be called when \c s3 is exited: \snippet statemachine/main.cpp 5 @@ -151,7 +151,7 @@ Assume we wanted the user to be able to quit the application at any time by clicking a Quit button. In order to achieve this, we need to create a final state and make it the target of a transition associated with the Quit - button's clicked() signal. We could add a transition from each of \c s1, \c + button's \l{QPushButton::}{clicked()} signal. We could add a transition from each of \c s1, \c s2 and \c s3; however, this seems redundant, and one would also have to remember to add such a transition from every new state that is added in the future. @@ -184,8 +184,8 @@ \snippet statemachine/main2.cpp 1 In this case we want the application to quit when the state machine is - finished, so the machine's finished() signal is connected to the - application's quit() slot. + finished, so the machine's \l {QStateMachine::}{finished()} signal is connected to the + application's \l {QCoreApplication::}{quit()} slot. A child state can override an inherited transition. For example, the following code adds a transition that effectively causes the Quit button to @@ -290,7 +290,7 @@ \endomit When \c s1 's final state is entered, \c s1 will automatically emit - finished(). We use a signal transition to cause this event to trigger a + \l {QState::}{finished()}. We use a signal transition to cause this event to trigger a state change: \snippet statemachine/main3.cpp 1 @@ -302,7 +302,7 @@ encapsulation mechanism when building complex (deeply nested) state machines. (In the above example, you could of course create a transition directly from \c s1 's \c done state rather than relying on \c s1 's - finished() signal, but with the consequence that implementation details of + \l {QState::}{finished()} signal, but with the consequence that implementation details of \c s1 are exposed and depended on). For parallel state groups, the QState::finished() signal is emitted when \e @@ -365,8 +365,8 @@ \snippet statemachine/main4.cpp 1 - In the eventTest() reimplementation, we first check if the event type is the - desired one; if so, we cast the event to a StringEvent and perform the + In the \l {QAbstractTransition::}{eventTest()} reimplementation, we first check if the event type is the + desired one; if so, we cast the event to a \c StringEvent and perform the string comparison. The following is a statechart that uses the custom event and transition: @@ -486,7 +486,7 @@ message box will pop up before the geometry of the button has actually been set. To ensure that the message box does not pop up until the geometry actually reaches its final - value, we can use the state's propertiesAssigned() signal. The propertiesAssigned() signal will be + value, we can use the state's \l {QState::}{propertiesAssigned()} signal. The \l {QState::}{propertiesAssigned()} signal will be emitted when the property is assigned its final value, whether this is done immediately or after the animation has finished playing. @@ -503,14 +503,14 @@ has been assigned the defined value. If the global restore policy is set to QStateMachine::RestoreProperties, the state will not emit - the propertiesAssigned() signal until these have been executed as well. + the \l {QState::}{propertiesAssigned()} signal until these have been executed as well. \section1 What Happens If A State Is Exited Before The Animation Has Finished If a state has property assignments, and the transition into the state has animations for the properties, the state can potentially be exited before the properties have been assigned to the values defines by the state. This is true in particular when there are transitions out from the - state that do not depend on the propertiesAssigned signal, as described in the previous section. + state that do not depend on the \l {QState::}{propertiesAssigned()} signal, as described in the previous section. The State Machine API guarantees that a property assigned by the state machine either: \list @@ -563,13 +563,13 @@ The parent state machine treats the child machine as an \e atomic state in the state machine algorithm. The child state machine is self-contained; it maintains its own event queue and - configuration. In particular, note that the \l{QStateMachine::configuration()}{configuration} + configuration. In particular, note that the \l{QStateMachine::}{configuration()} of the child machine is not part of the parent machine's configuration (only the child machine itself is). States of the child state machine cannot be specified as targets of transitions in the parent state machine; only the child state machine itself can. Conversely, states of the parent state machine cannot be specified as targets of transitions in the child state machine. The child - state machine's \l{QState::finished()}{finished}() signal can be used to trigger a transition + state machine's \l{QState::}{finished}() signal can be used to trigger a transition in the parent machine. */ From 083c9269ed73e8771e1dbe10812696b45b7389f3 Mon Sep 17 00:00:00 2001 From: Evangelos Foutras Date: Mon, 11 May 2015 12:20:57 +0300 Subject: [PATCH 08/24] Try to ensure that -fPIC is used in CMake builds In commit 36d6eb721e7d5997ade75e289d4088dc48678d0d the -fPIE switch was replaced with -fPIC in an effort to avoid generating copy relocations which are incompatible with Qt5 when built with -reduce-relocations. Task-number: QTBUG-45755 Change-Id: I59a55ea15052f498104848c5fd867e563ddc2290 Reviewed-by: Thiago Macieira --- src/corelib/Qt5CoreConfigExtras.cmake.in | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/src/corelib/Qt5CoreConfigExtras.cmake.in b/src/corelib/Qt5CoreConfigExtras.cmake.in index 48d5f21447..d4abc5f271 100644 --- a/src/corelib/Qt5CoreConfigExtras.cmake.in +++ b/src/corelib/Qt5CoreConfigExtras.cmake.in @@ -70,8 +70,9 @@ set(_qt5_corelib_extra_includes) # Qt5_POSITION_INDEPENDENT_CODE variable is used in the # qt5_use_module # macro to add it. set(Qt5_POSITION_INDEPENDENT_CODE True) -set_property(TARGET Qt5::Core PROPERTY INTERFACE_POSITION_INDEPENDENT_CODE \"ON\") set(Qt5Core_EXECUTABLE_COMPILE_FLAGS \"-fPIC\") +set_property(TARGET Qt5::Core PROPERTY INTERFACE_POSITION_INDEPENDENT_CODE \"ON\") +set_property(TARGET Qt5::Core APPEND PROPERTY INTERFACE_COMPILE_OPTIONS ${Qt5Core_EXECUTABLE_COMPILE_FLAGS}) !!IF !isEmpty(QT_NAMESPACE) list(APPEND Qt5Core_DEFINITIONS -DQT_NAMESPACE=$$QT_NAMESPACE) From 189280026f091716dfda91ff652cc20b299683c2 Mon Sep 17 00:00:00 2001 From: Richard Moe Gustavsen Date: Wed, 13 May 2015 14:33:10 +0200 Subject: [PATCH 09/24] xcode generator: use absolute path when creating PBXFileReferences for libraries MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Currently, the Xcode generator uses "sourceTree = " for all PBXFileReferences. But the paths we use for referencing libraries are relative. This patch will change this, so that we always use absolute paths to be consequent. This will fix a crash in Xcode that happens when opening projects generated by Qt. Change-Id: I3a372b93598a777c96ba353205cf19710a5923f5 Task-number: QTBUG-45966 Reviewed-by: Tor Arne Vestbø Reviewed-by: Oswald Buddenhagen --- qmake/generators/mac/pbuilder_pbx.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/qmake/generators/mac/pbuilder_pbx.cpp b/qmake/generators/mac/pbuilder_pbx.cpp index 8f24c20712..b54f634cf9 100644 --- a/qmake/generators/mac/pbuilder_pbx.cpp +++ b/qmake/generators/mac/pbuilder_pbx.cpp @@ -950,7 +950,7 @@ ProjectBuilderMakefileGenerator::writeMakeParts(QTextStream &t) if(!path.isEmpty() && !libdirs.contains(path)) libdirs += path; } - library = fileFixify(library); + library = fileFixify(library, FileFixifyAbsolute); QString filetype = xcodeFiletypeForFilename(library); QString key = keyFor(library); if (!project->values("QMAKE_PBX_LIBRARIES").contains(key)) { From 3f0f707d4b898a96e63e16c13a29f12be01d9b4c Mon Sep 17 00:00:00 2001 From: Mikhail Lappo Date: Thu, 7 May 2015 12:18:14 +0300 Subject: [PATCH 10/24] Prevent bad-ptrs deref in QNetworkConfigurationManagerPrivate Prevent application to crash with segfault in Qt bearer thread. Corrected hardly reproduceable bug, when QNetworkConfigurationManagerPrivate in pollEngines slot dereferenced null and bad pointers and caused crash Task-number: QTBUG-44407 Change-Id: I2f0b11b2d10125a21a62588d76ad824f375e4a1d Reviewed-by: Richard J. Moore --- src/network/bearer/qnetworkconfigmanager_p.cpp | 1 + 1 file changed, 1 insertion(+) diff --git a/src/network/bearer/qnetworkconfigmanager_p.cpp b/src/network/bearer/qnetworkconfigmanager_p.cpp index 6bbea1683c..b963aebbd5 100644 --- a/src/network/bearer/qnetworkconfigmanager_p.cpp +++ b/src/network/bearer/qnetworkconfigmanager_p.cpp @@ -75,6 +75,7 @@ QNetworkConfigurationManagerPrivate::~QNetworkConfigurationManagerPrivate() QMutexLocker locker(&mutex); qDeleteAll(sessionEngines); + sessionEngines.clear(); if (bearerThread) bearerThread->quit(); } From 20e36879d72f840005a0361368985b72ce4dc6a4 Mon Sep 17 00:00:00 2001 From: Venugopal Shivashankar Date: Fri, 15 May 2015 14:42:45 +0200 Subject: [PATCH 11/24] Doc: Excluded qdoc files that caused unnecessary Qt Creator warnings. MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit The customcompleter and textcodes are widget examples, but they end up in the Qt Core exampledirs boundary because of a reference to the plugandpaint example in the docs. This resulted in a couple of wrong entries being written into the examples-manifest.xml, which is used by Qt Creator. This change explicitly exludes the qdoc pages for the two examples so that qdoc doesn't add the corresponding entries into examples-manifest.xml. Task-number: QTBUG-41996 Change-Id: I0e95b6d4d93e0ce18f5b34e5034b279598b4924f Reviewed-by: Topi Reiniö Reviewed-by: Martin Smith --- src/corelib/doc/qtcore.qdocconf | 3 +++ 1 file changed, 3 insertions(+) diff --git a/src/corelib/doc/qtcore.qdocconf b/src/corelib/doc/qtcore.qdocconf index f3aff83a8b..a166df1143 100644 --- a/src/corelib/doc/qtcore.qdocconf +++ b/src/corelib/doc/qtcore.qdocconf @@ -42,5 +42,8 @@ imagedirs += images excludedirs += snippets +excludefiles += ../../../examples/widgets/tools/customcompleter/doc/src/customcompleter.qdoc \ + ../../../examples/widgets/tools/codecs/doc/src/codecs.qdoc + navigation.landingpage = "Qt Core" navigation.cppclassespage = "Qt Core C++ Classes" From 938f9fc0f8ff1575413ca3d6d66860b5fa315768 Mon Sep 17 00:00:00 2001 From: Paul Olav Tvete Date: Tue, 19 May 2015 16:07:42 +0200 Subject: [PATCH 12/24] Workaround for Samsung keyboard bug Return null string instead of empty string when the selection is empty. It looks like Samsung just tests for selection == null when doing backspace. Task-number: QTBUG-45785 Change-Id: Iaa006a8ffe52b2704c7348646dde9ca4e1f78c5c Reviewed-by: Eskil Abrahamsen Blomfeldt --- src/plugins/platforms/android/qandroidinputcontext.cpp | 2 ++ 1 file changed, 2 insertions(+) diff --git a/src/plugins/platforms/android/qandroidinputcontext.cpp b/src/plugins/platforms/android/qandroidinputcontext.cpp index 7e81735de9..7264f9a74c 100644 --- a/src/plugins/platforms/android/qandroidinputcontext.cpp +++ b/src/plugins/platforms/android/qandroidinputcontext.cpp @@ -176,6 +176,8 @@ static jstring getSelectedText(JNIEnv *env, jobject /*thiz*/, jint flags) #ifdef QT_DEBUG_ANDROID_IM_PROTOCOL qDebug() << "@@@ GETSEL" << text; #endif + if (text.isEmpty()) + return 0; return env->NewString(reinterpret_cast(text.constData()), jsize(text.length())); } From d4a296b538da93f5c915fc7e886c6995f1b41169 Mon Sep 17 00:00:00 2001 From: Allan Sandfeld Jensen Date: Wed, 20 May 2015 13:35:07 +0200 Subject: [PATCH 13/24] Do not modify decoder when determining image-format We should not configure the decoder when just determining the image- format. Doing so can cause all versions of libpng to print a warning, and some versions to fail to decode. The code appears to be a leftover from when the image-format logic was copied out of the introduction of the decoding method, where the proper settings are still applied. Task-number: QTBUG-46233 Change-Id: I6619728804f040ae6c9d637c7298a8586e22499e Reviewed-by: Andy Shaw Reviewed-by: aavit --- src/gui/image/qpnghandler.cpp | 7 ------- 1 file changed, 7 deletions(-) diff --git a/src/gui/image/qpnghandler.cpp b/src/gui/image/qpnghandler.cpp index 17a0dd3eb9..304ab0cf3e 100644 --- a/src/gui/image/qpnghandler.cpp +++ b/src/gui/image/qpnghandler.cpp @@ -674,16 +674,9 @@ QImage::Format QPngHandlerPrivate::readImageFormat() && num_palette <= 256) { // 1-bit and 8-bit color - if (bit_depth != 1) - png_set_packing(png_ptr); - png_read_update_info(png_ptr, info_ptr); - png_get_IHDR(png_ptr, info_ptr, &width, &height, &bit_depth, &color_type, 0, 0, 0); format = bit_depth == 1 ? QImage::Format_Mono : QImage::Format_Indexed8; } else { // 32-bit - if (bit_depth == 16) - png_set_strip_16(png_ptr); - format = QImage::Format_ARGB32; // Only add filler if no alpha, or we can get 5 channel data. if (!(color_type & PNG_COLOR_MASK_ALPHA) From e96ad10fd8f1d6badfe8ff7befbd397d5cbfe2b4 Mon Sep 17 00:00:00 2001 From: Andy Shaw Date: Wed, 20 May 2015 19:28:44 +0200 Subject: [PATCH 14/24] Ensure the same behavior on Win as Unix re the host in isApparentlyStale When the hostname is empty then it is assumed that the lock file is from the same host as the one running the application. Change-Id: Iba8aefc171a209294371dc2022d93ede3035b242 Reviewed-by: Will Wagner Reviewed-by: David Faure --- src/corelib/io/qlockfile_win.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/corelib/io/qlockfile_win.cpp b/src/corelib/io/qlockfile_win.cpp index 3587c7bffe..21ab8f8fea 100644 --- a/src/corelib/io/qlockfile_win.cpp +++ b/src/corelib/io/qlockfile_win.cpp @@ -120,7 +120,7 @@ bool QLockFilePrivate::isApparentlyStale() const // processes due to sandboxing #ifndef Q_OS_WINRT if (getLockInfo(&pid, &hostname, &appname)) { - if (hostname == QString::fromLocal8Bit(localHostName())) { + if (hostname.isEmpty() || hostname == QString::fromLocal8Bit(localHostName())) { HANDLE procHandle = ::OpenProcess(PROCESS_QUERY_INFORMATION, FALSE, pid); if (!procHandle) return true; From 7a4f3645f44bae78987facbf4e39231d0d2882ef Mon Sep 17 00:00:00 2001 From: "Richard J. Moore" Date: Sat, 23 May 2015 12:11:09 +0100 Subject: [PATCH 15/24] Avoid false positives from google by storing OpenSSL version as Unicode. Google scan play store apps for the openssl version string which leads to false positives since we record the version we were compiled against even though we don't link it directly. Task-number: QTBUG-46265 Change-Id: Iefd0e0954149c17350d49f57f9f374938124d7b8 Reviewed-by: Olivier Goffart (Woboq GmbH) --- src/network/ssl/qsslsocket_openssl.cpp | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/src/network/ssl/qsslsocket_openssl.cpp b/src/network/ssl/qsslsocket_openssl.cpp index b132aec038..ac4336afcc 100644 --- a/src/network/ssl/qsslsocket_openssl.cpp +++ b/src/network/ssl/qsslsocket_openssl.cpp @@ -588,7 +588,10 @@ long QSslSocketPrivate::sslLibraryBuildVersionNumber() QString QSslSocketPrivate::sslLibraryBuildVersionString() { - return QLatin1String(OPENSSL_VERSION_TEXT); + // Using QStringLiteral to store the version string as unicode and + // avoid false positives from Google searching the playstore for old + // SSL versions. See QTBUG-46265 + return QStringLiteral(OPENSSL_VERSION_TEXT); } /*! From ab156fcedd4c2d2856c0b46699c32926288be292 Mon Sep 17 00:00:00 2001 From: Andy Shaw Date: Fri, 17 Apr 2015 08:04:04 +0200 Subject: [PATCH 16/24] Use the new non-obsoleted functions for getting the paper details This fixes a problem with the margins not being correctly respected as the functions introduced previously would set a different property to what was being queried in this case. Change-Id: I3458c8e46239276a296d17aa80da7330c85fcf0a Reviewed-by: Friedemann Kleint --- src/printsupport/widgets/qprintpreviewwidget.cpp | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/printsupport/widgets/qprintpreviewwidget.cpp b/src/printsupport/widgets/qprintpreviewwidget.cpp index 208ad5e0f3..7898dc9352 100644 --- a/src/printsupport/widgets/qprintpreviewwidget.cpp +++ b/src/printsupport/widgets/qprintpreviewwidget.cpp @@ -335,8 +335,8 @@ void QPrintPreviewWidgetPrivate::populateScene() pages.clear(); int numPages = pictures.count(); - QSize paperSize = printer->paperRect().size(); - QRect pageRect = printer->pageRect(); + QSize paperSize = printer->pageLayout().fullRectPixels(printer->resolution()).size(); + QRect pageRect = printer->pageLayout().paintRectPixels(printer->resolution()); for (int i = 0; i < numPages; i++) { PageItem* item = new PageItem(i+1, pictures.at(i), paperSize, pageRect); From 79be2601225bdb50cab72e2502b7d7f9ee81e94f Mon Sep 17 00:00:00 2001 From: Andy Shaw Date: Fri, 17 Apr 2015 08:10:10 +0200 Subject: [PATCH 17/24] Fix documentation of obsoleted functions to show the right replacement Change-Id: Ib008a5544d68d93e1f96ff6b7504e9a7ea4bb192 Reviewed-by: Friedemann Kleint --- src/printsupport/kernel/qprinter.cpp | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/printsupport/kernel/qprinter.cpp b/src/printsupport/kernel/qprinter.cpp index 8ed2732c1e..d7df796afb 100644 --- a/src/printsupport/kernel/qprinter.cpp +++ b/src/printsupport/kernel/qprinter.cpp @@ -1764,7 +1764,7 @@ QRectF QPrinter::paperRect(Unit unit) const } /*! - \obsolete Use pageLayout().paintRect() instead. + \obsolete Use pageLayout().paintRectPixels(resolution()) instead. Returns the page's rectangle; this is usually smaller than the paperRect() since the page normally has margins between its @@ -1781,7 +1781,7 @@ QRect QPrinter::pageRect() const } /*! - \obsolete Use pageLayout().fullPageRect() instead. + \obsolete Use pageLayout().fullRectPixels(resolution()) instead. Returns the paper's rectangle; this is usually larger than the pageRect(). From 18be63de2a5588ea6ac4b9ffe8de176aabe8bd83 Mon Sep 17 00:00:00 2001 From: Venugopal Shivashankar Date: Mon, 18 May 2015 15:47:51 +0200 Subject: [PATCH 18/24] Doc: Added the missing \brief and \image for example docs MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit The Qt TestLib examples are just tutorials so updated the qdocconf to use the default thumbnail in the Qt Creator welcome screen. Task-number: QTBUG-41996 Change-Id: Ia04a42a92e414c97a426b6095a62621a348e7de0 Reviewed-by: Topi Reiniö Reviewed-by: Martin Smith --- .../doc/images/imagegestures-example.jpg | Bin 0 -> 38962 bytes examples/widgets/doc/src/plugandpaint.qdoc | 9 +++++++-- .../imagegestures/doc/src/imagegestures.qdoc | 3 +++ src/testlib/doc/qttestlib.qdocconf | 3 +++ src/xml/doc/qtxml.qdocconf | 3 +++ 5 files changed, 16 insertions(+), 2 deletions(-) create mode 100644 examples/widgets/doc/images/imagegestures-example.jpg diff --git a/examples/widgets/doc/images/imagegestures-example.jpg b/examples/widgets/doc/images/imagegestures-example.jpg new file mode 100644 index 0000000000000000000000000000000000000000..c8484b4a71999b9f98e449a3a0a48c951e9f8253 GIT binary patch literal 38962 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 h6Xt~b93K2`p2>VH~DZzBKz literal 0 HcmV?d00001 diff --git a/examples/widgets/doc/src/plugandpaint.qdoc b/examples/widgets/doc/src/plugandpaint.qdoc index 4e48245bd9..caf33feb0a 100644 --- a/examples/widgets/doc/src/plugandpaint.qdoc +++ b/examples/widgets/doc/src/plugandpaint.qdoc @@ -30,8 +30,7 @@ \title Plug & Paint Example \ingroup examples-widgets-tools - \brief The Plug & Paint example demonstrates how to write Qt - applications that can be extended through plugins. + \brief Demonstrates how to extend Qt applications using plugins. \image plugandpaint.png Screenshot of the Plug & Paint example @@ -314,6 +313,9 @@ /*! \example tools/plugandpaintplugins/basictools \title Plug & Paint Basic Tools Example + \brief A plugin providing the basic tools for painting functionality. + + \image plugandpaint.png Screenshot of the Plug & Paint example The Basic Tools example is a static plugin for the \l{tools/plugandpaint}{Plug & Paint} example. It provides a set @@ -499,6 +501,9 @@ /*! \example tools/plugandpaintplugins/extrafilters \title Plug & Paint Extra Filters Example + \brief A plugin providing the extra filters. + + \image plugandpaint.png Screenshot of the Plug & Paint example The Extra Filters example is a plugin for the \l{tools/plugandpaint}{Plug & Paint} example. It provides a set diff --git a/examples/widgets/gestures/imagegestures/doc/src/imagegestures.qdoc b/examples/widgets/gestures/imagegestures/doc/src/imagegestures.qdoc index d6d215d6bd..6a1405b0b1 100644 --- a/examples/widgets/gestures/imagegestures/doc/src/imagegestures.qdoc +++ b/examples/widgets/gestures/imagegestures/doc/src/imagegestures.qdoc @@ -28,10 +28,13 @@ /*! \example gestures/imagegestures \title Image Gestures Example + \brief Demonstrates the use of simple gestures in a widget This example shows how to enable gestures for a widget and use gesture input to perform actions. + \image imagegestures-example.jpg + We use two classes to create the user interface for the application: \c MainWidget and \c ImageWidget. The \c MainWidget class is simply used as a container for the \c ImageWidget class, which we will configure to accept gesture input. Since we diff --git a/src/testlib/doc/qttestlib.qdocconf b/src/testlib/doc/qttestlib.qdocconf index 35b4fbcb7b..0fafc733b1 100644 --- a/src/testlib/doc/qttestlib.qdocconf +++ b/src/testlib/doc/qttestlib.qdocconf @@ -40,5 +40,8 @@ excludedirs += ../../../examples/widgets/doc imagedirs += images +# Add a thumbnail for examples that do not have images +manifestmeta.thumbnail.names = "QtTestLib/Chapter *" + navigation.landingpage = "Qt Test" navigation.cppclassespage = "Qt Test C++ Classes" diff --git a/src/xml/doc/qtxml.qdocconf b/src/xml/doc/qtxml.qdocconf index 8ca421ff4e..419859ac8b 100644 --- a/src/xml/doc/qtxml.qdocconf +++ b/src/xml/doc/qtxml.qdocconf @@ -40,3 +40,6 @@ imagedirs += images \ navigation.landingpage = "Qt XML" navigation.cppclassespage = "Qt XML C++ Classes" + +# Add a thumbnail for examples that do not have images +manifestmeta.thumbnail.names = "QtXml/XML Stream Lint Example" From 8829ce67d8d0eac90e9fd6fde088b41f157177a5 Mon Sep 17 00:00:00 2001 From: Stephen Kelly Date: Mon, 25 May 2015 14:55:38 +0200 Subject: [PATCH 19/24] Only add -fPIC flags for compilers known to require it. Commit 083c9269 (Try to ensure that -fPIC is used in CMake builds, 2015-05-11) added a raw -fPIC to the INTERFACE_COMPILE_OPTIONS of Qt5::Core, which affects all consuming compilers. Use the qmake variable $$QMAKE_CXXFLAGS_APP instead, which at least currently contains only the -fPIC variable or harmlessly expands to nothing. If the content of that qmake variable changes in the future, a $$QMAKE_CXXFLAGS_APP_PIC variable should be extracted in qmake and used here. Don't use the POSITION_INDEPENDENT_CODE feature of CMake. That adds the -fPIE flag for executables, which is explicitly what qglobal.h forbids since commit 3eca75de (Make qglobal.h complain if you use -fPIE, 2015-05-11). The current behavior of that CMake feature is tracked here: http://public.kitware.com/Bug/view.php?id=15570 Change-Id: I5c5bcc40fe4b310b55a681a3505f45c50adfa054 Reviewed-by: Oswald Buddenhagen Reviewed-by: Simon Hausmann --- src/corelib/Qt5CoreConfigExtras.cmake.in | 3 +-- src/corelib/Qt5CoreMacros.cmake | 4 ---- 2 files changed, 1 insertion(+), 6 deletions(-) diff --git a/src/corelib/Qt5CoreConfigExtras.cmake.in b/src/corelib/Qt5CoreConfigExtras.cmake.in index d4abc5f271..a5cab880ba 100644 --- a/src/corelib/Qt5CoreConfigExtras.cmake.in +++ b/src/corelib/Qt5CoreConfigExtras.cmake.in @@ -71,8 +71,7 @@ set(_qt5_corelib_extra_includes) # macro to add it. set(Qt5_POSITION_INDEPENDENT_CODE True) set(Qt5Core_EXECUTABLE_COMPILE_FLAGS \"-fPIC\") -set_property(TARGET Qt5::Core PROPERTY INTERFACE_POSITION_INDEPENDENT_CODE \"ON\") -set_property(TARGET Qt5::Core APPEND PROPERTY INTERFACE_COMPILE_OPTIONS ${Qt5Core_EXECUTABLE_COMPILE_FLAGS}) +set_property(TARGET Qt5::Core APPEND PROPERTY INTERFACE_COMPILE_OPTIONS $$QMAKE_CXXFLAGS_APP) !!IF !isEmpty(QT_NAMESPACE) list(APPEND Qt5Core_DEFINITIONS -DQT_NAMESPACE=$$QT_NAMESPACE) diff --git a/src/corelib/Qt5CoreMacros.cmake b/src/corelib/Qt5CoreMacros.cmake index 9c81754302..c10880f787 100644 --- a/src/corelib/Qt5CoreMacros.cmake +++ b/src/corelib/Qt5CoreMacros.cmake @@ -281,10 +281,6 @@ if (NOT CMAKE_VERSION VERSION_LESS 2.8.9) set_property(TARGET ${_target} APPEND PROPERTY COMPILE_DEFINITIONS_RELEASE QT_NO_DEBUG) set_property(TARGET ${_target} APPEND PROPERTY COMPILE_DEFINITIONS_RELWITHDEBINFO QT_NO_DEBUG) set_property(TARGET ${_target} APPEND PROPERTY COMPILE_DEFINITIONS_MINSIZEREL QT_NO_DEBUG) - - if (Qt5_POSITION_INDEPENDENT_CODE) - set_property(TARGET ${_target} PROPERTY POSITION_INDEPENDENT_CODE ${Qt5_POSITION_INDEPENDENT_CODE}) - endif() endforeach() endmacro() endif() From 6255859ebf19b38c864cd6dbaf2878e0116535a4 Mon Sep 17 00:00:00 2001 From: Simon Hausmann Date: Wed, 27 May 2015 16:17:19 +0200 Subject: [PATCH 20/24] Fix cmake auto tests failing with cmake 2.8.11 MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit After commit 3eca75de67b3fd2c890715b30c7899cebc096fe9 the use of _EXECUTABLE_COMPILE_FLAGS becomes mandatory for 2.8.11 or older. Therefore use it in test_use_modules - that's supposed to work with all cmake versions it seems - and bump test_interface to require 2.8.12. Change-Id: I7cfb6c6f1e8a97be916d372b9d9148490926693c Reviewed-by: Lisandro Damián Nicanor Pérez Meyer Reviewed-by: Stephen Kelly --- tests/auto/cmake/CMakeLists.txt | 5 +---- tests/auto/cmake/test_interface/CMakeLists.txt | 2 +- tests/auto/cmake/test_use_modules_function/CMakeLists.txt | 2 ++ 3 files changed, 4 insertions(+), 5 deletions(-) diff --git a/tests/auto/cmake/CMakeLists.txt b/tests/auto/cmake/CMakeLists.txt index 87d8a802f3..b68aafe956 100644 --- a/tests/auto/cmake/CMakeLists.txt +++ b/tests/auto/cmake/CMakeLists.txt @@ -126,11 +126,8 @@ if (QT_WITH_ANGLE OR (NOT WIN32 AND NOT APPLE AND NOT NO_EGL)) endif() expect_pass(test_opengl_lib) -if (NOT CMAKE_VERSION VERSION_LESS 2.8.11) - expect_pass(test_interface) -endif() - if (NOT CMAKE_VERSION VERSION_LESS 2.8.12) + expect_pass(test_interface) expect_pass(test_interface_link_libraries) expect_pass(test_moc_macro_target) endif() diff --git a/tests/auto/cmake/test_interface/CMakeLists.txt b/tests/auto/cmake/test_interface/CMakeLists.txt index a8af92da63..bd3217a497 100644 --- a/tests/auto/cmake/test_interface/CMakeLists.txt +++ b/tests/auto/cmake/test_interface/CMakeLists.txt @@ -1,5 +1,5 @@ -cmake_minimum_required(VERSION 2.8.11) +cmake_minimum_required(VERSION 2.8.12) project(test_interface) diff --git a/tests/auto/cmake/test_use_modules_function/CMakeLists.txt b/tests/auto/cmake/test_use_modules_function/CMakeLists.txt index bfcdd9d1d7..be05c75054 100644 --- a/tests/auto/cmake/test_use_modules_function/CMakeLists.txt +++ b/tests/auto/cmake/test_use_modules_function/CMakeLists.txt @@ -12,5 +12,7 @@ add_executable(three three.cpp) find_package(Qt5Core) +set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} ${Qt5Core_EXECUTABLE_COMPILE_FLAGS}") + qt5_use_modules(two Test) qt5_use_modules(three Gui Test) From 9d662d9d2f33d75d3fbf682ee9fadc291ecae5cb Mon Sep 17 00:00:00 2001 From: Simon Hausmann Date: Wed, 27 May 2015 12:47:12 +0200 Subject: [PATCH 21/24] Mention cmake changes in ChangeLog for -fPIE -> -fPIC changes This is a follow-up change to commit 3eca75de67b3fd2c890715b30c7899cebc096fe9 to mention changes required to CMakeLists.txt for users of Qt. Change-Id: I1c9ed162427cdc620f998ccf266d59886901c28d Reviewed-by: Lars Knoll --- dist/changes-5.4.2 | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/dist/changes-5.4.2 b/dist/changes-5.4.2 index 5827187e9e..29d3c6346d 100644 --- a/dist/changes-5.4.2 +++ b/dist/changes-5.4.2 @@ -41,6 +41,10 @@ information about a particular change. Qt's option "reduce relocations" is active. Note that Clang is known to generate incompatible code even with -fPIC if the -flto option is active. + Applications using qmake or cmake >= 2.8.12 as their build system will + adapt automatically. Applications using an older release of cmake need to + change their CMakeLists.txt to add Qt5Core_EXECUTABLE_COMPILE_FLAGS to + CMAKE_CXX_FLAGS. **************************************************************************** * Library * From 95b6c4fed6521aa2212cab67cb8a6e5553e86117 Mon Sep 17 00:00:00 2001 From: Thiago Macieira Date: Thu, 28 May 2015 11:21:35 -0700 Subject: [PATCH 22/24] Make qglobal.h only complain for GCC >= 5 about -fPIE Commit 3eca75de67b3fd2c890715b30c7899cebc096fe9 introduced the #error nagging about use of -fPIE, but it makes the transition quite difficult for people using other buildsystems. So let's give people a grace period and enforce only for GCC >= 5. Clang is affected, but differently. The problem only happens with -flto -- that is, it happens when the linker detects that it's creating a final executable. Maybe -Wl,-pie would fix it. Change-Id: If4d5ac8db0ed4a84a3eaffff13e275edc29a72b7 Reviewed-by: Simon Hausmann Reviewed-by: Dmitry Shachnev --- src/corelib/global/qglobal.h | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/src/corelib/global/qglobal.h b/src/corelib/global/qglobal.h index 4547877da6..d9742408a3 100644 --- a/src/corelib/global/qglobal.h +++ b/src/corelib/global/qglobal.h @@ -1047,7 +1047,8 @@ Q_CORE_EXPORT int qrand(); # define QT_NO_SHAREDMEMORY #endif -#if !defined(QT_BOOTSTRAPPED) && defined(QT_REDUCE_RELOCATIONS) && defined(__ELF__) && (!defined(__PIC__) || defined(__PIE__)) +#if !defined(QT_BOOTSTRAPPED) && defined(QT_REDUCE_RELOCATIONS) && defined(__ELF__) && \ + (!defined(__PIC__) || (defined(__PIE__) && defined(Q_CC_GNU) && Q_CC_GNU >= 500)) # error "You must build your code with position independent code if Qt was built with -reduce-relocations. "\ "Compile your code with -fPIC (-fPIE is not enough)." #endif From e3983c87280ade48b243d9c60bed639713851be9 Mon Sep 17 00:00:00 2001 From: Stephen Kelly Date: Thu, 28 May 2015 21:20:55 +0200 Subject: [PATCH 23/24] Revert some changes in light of GCC 4 -fPIE reversal The -fPIE option is now accepted when using GCC 4, which means it is available for backward compatibility for clients using CMake 2.8.11 or older which makes use of the POSITION_INDEPENDENT_CODE feature. Conditionally use that feature for old versions of cmake with GCC 4. Restore the tests for those versions, and clarify the situation in the ChangeLog. Change-Id: I5a06b155dda7db559d86841a2b34fd8ed95acbd0 Reviewed-by: Thiago Macieira --- dist/changes-5.4.2 | 20 ++++++++++++------- src/corelib/Qt5CoreConfigExtras.cmake.in | 8 +++++++- src/corelib/Qt5CoreMacros.cmake | 6 ++++++ tests/auto/cmake/CMakeLists.txt | 5 ++++- .../auto/cmake/test_interface/CMakeLists.txt | 2 +- .../test_use_modules_function/CMakeLists.txt | 2 -- 6 files changed, 31 insertions(+), 12 deletions(-) diff --git a/dist/changes-5.4.2 b/dist/changes-5.4.2 index 29d3c6346d..1a34898011 100644 --- a/dist/changes-5.4.2 +++ b/dist/changes-5.4.2 @@ -37,14 +37,20 @@ information about a particular change. - On x86 and x86-64 systems with ELF binaries (especially Linux), due to a new optimization in GCC 5.x in combination with a recent version of GNU binutils, compiling Qt applications with -fPIE is no longer - enough. Applications now need to be compiled with the -fPIC option if - Qt's option "reduce relocations" is active. Note that Clang is known - to generate incompatible code even with -fPIC if the -flto option is - active. + enough with GCC 5.x. Applications now need to be compiled with + the -fPIC option if Qt's option "reduce relocations" is active. For + backward compatibility only, Qt accepts the use of -fPIE for GCC 4.x + versions. + Note that Clang is known to generate incompatible code even with -fPIC if + the -flto option is active. Applications using qmake or cmake >= 2.8.12 as their build system will - adapt automatically. Applications using an older release of cmake need to - change their CMakeLists.txt to add Qt5Core_EXECUTABLE_COMPILE_FLAGS to - CMAKE_CXX_FLAGS. + adapt automatically. Applications using an older release of cmake in + combination with GCC 5.x need to change their CMakeLists.txt to add + Qt5Core_EXECUTABLE_COMPILE_FLAGS to CMAKE_CXX_FLAGS. In particular, + applications using cmake >= 2.8.9 and < 2.8.11 will continue to build + with the -fPIE option and invoke the special compatibility mode if using + GCC 4.x. + **************************************************************************** * Library * diff --git a/src/corelib/Qt5CoreConfigExtras.cmake.in b/src/corelib/Qt5CoreConfigExtras.cmake.in index a5cab880ba..65fd1f9383 100644 --- a/src/corelib/Qt5CoreConfigExtras.cmake.in +++ b/src/corelib/Qt5CoreConfigExtras.cmake.in @@ -71,7 +71,13 @@ set(_qt5_corelib_extra_includes) # macro to add it. set(Qt5_POSITION_INDEPENDENT_CODE True) set(Qt5Core_EXECUTABLE_COMPILE_FLAGS \"-fPIC\") -set_property(TARGET Qt5::Core APPEND PROPERTY INTERFACE_COMPILE_OPTIONS $$QMAKE_CXXFLAGS_APP) +if (CMAKE_VERSION VERSION_LESS 2.8.12 + AND (NOT CMAKE_CXX_COMPILER_ID STREQUAL \"GNU\" + OR CMAKE_CXX_COMPILER_VERSION VERSION_LESS 5.0)) + set_property(TARGET Qt5::Core APPEND PROPERTY INTERFACE_POSITION_INDEPENDENT_CODE \"ON\") +else() + set_property(TARGET Qt5::Core APPEND PROPERTY INTERFACE_COMPILE_OPTIONS $$QMAKE_CXXFLAGS_APP) +endif() !!IF !isEmpty(QT_NAMESPACE) list(APPEND Qt5Core_DEFINITIONS -DQT_NAMESPACE=$$QT_NAMESPACE) diff --git a/src/corelib/Qt5CoreMacros.cmake b/src/corelib/Qt5CoreMacros.cmake index c10880f787..cfbb381df5 100644 --- a/src/corelib/Qt5CoreMacros.cmake +++ b/src/corelib/Qt5CoreMacros.cmake @@ -281,6 +281,12 @@ if (NOT CMAKE_VERSION VERSION_LESS 2.8.9) set_property(TARGET ${_target} APPEND PROPERTY COMPILE_DEFINITIONS_RELEASE QT_NO_DEBUG) set_property(TARGET ${_target} APPEND PROPERTY COMPILE_DEFINITIONS_RELWITHDEBINFO QT_NO_DEBUG) set_property(TARGET ${_target} APPEND PROPERTY COMPILE_DEFINITIONS_MINSIZEREL QT_NO_DEBUG) + if (Qt5_POSITION_INDEPENDENT_CODE + AND (CMAKE_VERSION VERSION_LESS 2.8.12 + AND (NOT CMAKE_CXX_COMPILER_ID STREQUAL \"GNU\" + OR CMAKE_CXX_COMPILER_VERSION VERSION_LESS 5.0))) + set_property(TARGET ${_target} PROPERTY POSITION_INDEPENDENT_CODE ${Qt5_POSITION_INDEPENDENT_CODE}) + endif() endforeach() endmacro() endif() diff --git a/tests/auto/cmake/CMakeLists.txt b/tests/auto/cmake/CMakeLists.txt index b68aafe956..87d8a802f3 100644 --- a/tests/auto/cmake/CMakeLists.txt +++ b/tests/auto/cmake/CMakeLists.txt @@ -126,8 +126,11 @@ if (QT_WITH_ANGLE OR (NOT WIN32 AND NOT APPLE AND NOT NO_EGL)) endif() expect_pass(test_opengl_lib) -if (NOT CMAKE_VERSION VERSION_LESS 2.8.12) +if (NOT CMAKE_VERSION VERSION_LESS 2.8.11) expect_pass(test_interface) +endif() + +if (NOT CMAKE_VERSION VERSION_LESS 2.8.12) expect_pass(test_interface_link_libraries) expect_pass(test_moc_macro_target) endif() diff --git a/tests/auto/cmake/test_interface/CMakeLists.txt b/tests/auto/cmake/test_interface/CMakeLists.txt index bd3217a497..a8af92da63 100644 --- a/tests/auto/cmake/test_interface/CMakeLists.txt +++ b/tests/auto/cmake/test_interface/CMakeLists.txt @@ -1,5 +1,5 @@ -cmake_minimum_required(VERSION 2.8.12) +cmake_minimum_required(VERSION 2.8.11) project(test_interface) diff --git a/tests/auto/cmake/test_use_modules_function/CMakeLists.txt b/tests/auto/cmake/test_use_modules_function/CMakeLists.txt index be05c75054..bfcdd9d1d7 100644 --- a/tests/auto/cmake/test_use_modules_function/CMakeLists.txt +++ b/tests/auto/cmake/test_use_modules_function/CMakeLists.txt @@ -12,7 +12,5 @@ add_executable(three three.cpp) find_package(Qt5Core) -set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} ${Qt5Core_EXECUTABLE_COMPILE_FLAGS}") - qt5_use_modules(two Test) qt5_use_modules(three Gui Test) From 386aca1ba4da3383cd6b6253a7240417ff2d91a0 Mon Sep 17 00:00:00 2001 From: Lorn Potter Date: Fri, 22 May 2015 09:13:51 +1000 Subject: [PATCH 24/24] Fix no bearermanagement build As pointed out in the bug, it also fixes API use when configured with no bearermanagement. Task-number: QTBUG-46239 Change-Id: Ief8df85ad6acf61e8d5bb3eed54e7d6ecb84c1a0 Reviewed-by: Alex Blasche --- src/network/access/qnetworkaccessmanager.cpp | 41 ++++++++++--------- .../bearer/corewlan/qcorewlanengine.mm | 3 ++ 2 files changed, 24 insertions(+), 20 deletions(-) diff --git a/src/network/access/qnetworkaccessmanager.cpp b/src/network/access/qnetworkaccessmanager.cpp index 52d56fb071..14db4554bb 100644 --- a/src/network/access/qnetworkaccessmanager.cpp +++ b/src/network/access/qnetworkaccessmanager.cpp @@ -961,6 +961,27 @@ QNetworkAccessManager::NetworkAccessibility QNetworkAccessManager::networkAccess } } +/*! + \internal + + Returns the network session currently in use. + This can be changed at any time, ownership remains with the QNetworkAccessManager +*/ +const QWeakPointer QNetworkAccessManagerPrivate::getNetworkSession(const QNetworkAccessManager *q) +{ + return q->d_func()->networkSessionWeakRef; +} + +QSharedPointer QNetworkAccessManagerPrivate::getNetworkSession() const +{ + if (networkSessionStrongRef) + return networkSessionStrongRef; + return networkSessionWeakRef.toStrongRef(); +} + +#endif // QT_NO_BEARERMANAGEMENT + + #ifndef QT_NO_SSL /*! \since 5.2 @@ -1021,26 +1042,6 @@ void QNetworkAccessManager::connectToHost(const QString &hostName, quint16 port) get(request); } -/*! - \internal - - Returns the network session currently in use. - This can be changed at any time, ownership remains with the QNetworkAccessManager -*/ -const QWeakPointer QNetworkAccessManagerPrivate::getNetworkSession(const QNetworkAccessManager *q) -{ - return q->d_func()->networkSessionWeakRef; -} - -QSharedPointer QNetworkAccessManagerPrivate::getNetworkSession() const -{ - if (networkSessionStrongRef) - return networkSessionStrongRef; - return networkSessionWeakRef.toStrongRef(); -} - -#endif // QT_NO_BEARERMANAGEMENT - /*! \since 4.7 diff --git a/src/plugins/bearer/corewlan/qcorewlanengine.mm b/src/plugins/bearer/corewlan/qcorewlanengine.mm index 2b38409723..3cee70044f 100644 --- a/src/plugins/bearer/corewlan/qcorewlanengine.mm +++ b/src/plugins/bearer/corewlan/qcorewlanengine.mm @@ -52,6 +52,7 @@ #include #include +#ifndef QT_NO_BEARERMANAGEMENT extern "C" { // Otherwise it won't find CWKeychain* symbols at link time #import @@ -896,3 +897,5 @@ quint64 QCoreWlanEngine::getBytes(const QString &interfaceName, bool b) } QT_END_NAMESPACE + +#endif