summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorVolker Hilsheimer <volker.hilsheimer@qt.io>2023-10-16 17:21:08 +0200
committerVolker Hilsheimer <volker.hilsheimer@qt.io>2023-10-18 15:02:22 +0200
commitb7a7351767bc58bc7e8719972e309a1e9f23967c (patch)
treeda52fd46f22c1c07e3bca43f02bedafb2c5bb038
parent872f7b0b4e7ddfbbe96612c4470466681df41aba (diff)
JNI: treat equivalent C++ types as the same JNI types as well
Our signature mapping treats both e.g. bool and jboolean as "Z", and it is allowed to pass a bool variable as an argument to a function expecting a jboolean. Except for fields and callMethod return values, where we only allowed the JNI primitive types. Fix this by comparing the signatures and size of the type we have with the JNI types that there are explicit functions for. Cast from and to the JNI type in both directions to address narrowing (e.g. jboolean is an unsigned char and converting to bool would be narrowing, even though both are 8bit types). This way we can get boolean fields using getField<bool>, and int fields using getField<int> etc. Change-Id: I2f1ba855ee01423e79ba999dfb9d86f4b98b1402 Reviewed-by: Qt CI Bot <qt_ci_bot@qt-project.org> Reviewed-by: Assam Boudjelthia <assam.boudjelthia@qt.io>
-rw-r--r--src/corelib/kernel/qjniobject.h190
-rw-r--r--tests/auto/corelib/kernel/qjniobject/tst_qjniobject.cpp10
2 files changed, 100 insertions, 100 deletions
diff --git a/src/corelib/kernel/qjniobject.h b/src/corelib/kernel/qjniobject.h
index 05db6f4414..88214a9311 100644
--- a/src/corelib/kernel/qjniobject.h
+++ b/src/corelib/kernel/qjniobject.h
@@ -561,20 +561,10 @@ private:
friend bool operator==(const QJniObject &, const QJniObject &);
friend bool operator!=(const QJniObject&, const QJniObject&);
- template<typename T, typename = void>
- struct IntegerTypeDetail
- {
- using type = T;
- };
- template<typename T>
- struct IntegerTypeDetail<T, typename std::enable_if_t<std::is_enum_v<T>>>
- {
- using type = std::underlying_type_t<T>;
- };
-
- template<typename Have, typename Want>
- static constexpr bool likeIntegerType = std::is_same_v<Have, Want>
- || std::is_same_v<typename IntegerTypeDetail<Have>::type, Want>;
+ template <typename Have, typename Want>
+ static constexpr bool sameTypeForJni = (QtJniTypes::Traits<Have>::signature()
+ == QtJniTypes::Traits<Want>::signature())
+ && (sizeof(Have) == sizeof(Want));
template<typename T>
static constexpr void callMethodForType(JNIEnv *env, T &res, jobject obj,
@@ -583,22 +573,22 @@ private:
va_list args = {};
va_start(args, id);
- if constexpr (std::is_same_v<T, jboolean>)
- res = env->CallBooleanMethodV(obj, id, args);
- else if constexpr (likeIntegerType<T, jbyte>)
- res = T{env->CallByteMethodV(obj, id, args)};
- else if constexpr (likeIntegerType<T, jchar>)
- res = T{env->CallCharMethodV(obj, id, args)};
- else if constexpr (likeIntegerType<T, jshort>)
- res = T{env->CallShortMethodV(obj, id, args)};
- else if constexpr (likeIntegerType<T, jint>)
- res = T{env->CallIntMethodV(obj, id, args)};
- else if constexpr (likeIntegerType<T, jlong>)
- res = T{env->CallLongMethodV(obj, id, args)};
- else if constexpr (std::is_same_v<T, jfloat>)
- res = env->CallFloatMethodV(obj, id, args);
- else if constexpr (std::is_same_v<T, jdouble>)
- res = env->CallDoubleMethodV(obj, id, args);
+ if constexpr (sameTypeForJni<T, jboolean>)
+ res = T(env->CallBooleanMethodV(obj, id, args));
+ else if constexpr (sameTypeForJni<T, jbyte>)
+ res = T(env->CallByteMethodV(obj, id, args));
+ else if constexpr (sameTypeForJni<T, jchar>)
+ res = T(env->CallCharMethodV(obj, id, args));
+ else if constexpr (sameTypeForJni<T, jshort>)
+ res = T(env->CallShortMethodV(obj, id, args));
+ else if constexpr (sameTypeForJni<T, jint>)
+ res = T(env->CallIntMethodV(obj, id, args));
+ else if constexpr (sameTypeForJni<T, jlong>)
+ res = T(env->CallLongMethodV(obj, id, args));
+ else if constexpr (sameTypeForJni<T, jfloat>)
+ res = T(env->CallFloatMethodV(obj, id, args));
+ else if constexpr (sameTypeForJni<T, jdouble>)
+ res = T(env->CallDoubleMethodV(obj, id, args));
else
QtJniTypes::staticAssertTypeMismatch();
va_end(args);
@@ -612,22 +602,22 @@ private:
return;
va_list args = {};
va_start(args, id);
- if constexpr (std::is_same_v<T, jboolean>)
- res = env->CallStaticBooleanMethodV(clazz, id, args);
- else if constexpr (likeIntegerType<T, jbyte>)
- res = T{env->CallStaticByteMethodV(clazz, id, args)};
- else if constexpr (likeIntegerType<T, jchar>)
- res = T{env->CallStaticCharMethodV(clazz, id, args)};
- else if constexpr (likeIntegerType<T, jshort>)
- res = T{env->CallStaticShortMethodV(clazz, id, args)};
- else if constexpr (likeIntegerType<T, jint>)
- res = T{env->CallStaticIntMethodV(clazz, id, args)};
- else if constexpr (likeIntegerType<T, jlong>)
- res = T{env->CallStaticLongMethodV(clazz, id, args)};
- else if constexpr (std::is_same_v<T, jfloat>)
- res = env->CallStaticFloatMethodV(clazz, id, args);
- else if constexpr (std::is_same_v<T, jdouble>)
- res = env->CallStaticDoubleMethodV(clazz, id, args);
+ if constexpr (sameTypeForJni<T, jboolean>)
+ res = T(env->CallStaticBooleanMethodV(clazz, id, args));
+ else if constexpr (sameTypeForJni<T, jbyte>)
+ res = T(env->CallStaticByteMethodV(clazz, id, args));
+ else if constexpr (sameTypeForJni<T, jchar>)
+ res = T(env->CallStaticCharMethodV(clazz, id, args));
+ else if constexpr (sameTypeForJni<T, jshort>)
+ res = T(env->CallStaticShortMethodV(clazz, id, args));
+ else if constexpr (sameTypeForJni<T, jint>)
+ res = T(env->CallStaticIntMethodV(clazz, id, args));
+ else if constexpr (sameTypeForJni<T, jlong>)
+ res = T(env->CallStaticLongMethodV(clazz, id, args));
+ else if constexpr (sameTypeForJni<T, jfloat>)
+ res = T(env->CallStaticFloatMethodV(clazz, id, args));
+ else if constexpr (sameTypeForJni<T, jdouble>)
+ res = T(env->CallStaticDoubleMethodV(clazz, id, args));
else
QtJniTypes::staticAssertTypeMismatch();
va_end(args);
@@ -648,22 +638,22 @@ private:
static constexpr void getFieldForType(JNIEnv *env, T &res, jobject obj,
jfieldID id)
{
- if constexpr (std::is_same_v<T, jboolean>)
- res = env->GetBooleanField(obj, id);
- else if constexpr (likeIntegerType<T, jbyte>)
- res = T{env->GetByteField(obj, id)};
- else if constexpr (likeIntegerType<T, jchar>)
- res = T{env->GetCharField(obj, id)};
- else if constexpr (likeIntegerType<T, jshort>)
- res = T{env->GetShortField(obj, id)};
- else if constexpr (likeIntegerType<T, jint>)
- res = T{env->GetIntField(obj, id)};
- else if constexpr (likeIntegerType<T, jlong>)
- res = T{env->GetLongField(obj, id)};
- else if constexpr (std::is_same_v<T, jfloat>)
- res = env->GetFloatField(obj, id);
- else if constexpr (std::is_same_v<T, jdouble>)
- res = env->GetDoubleField(obj, id);
+ if constexpr (sameTypeForJni<T, jboolean>)
+ res = T(env->GetBooleanField(obj, id));
+ else if constexpr (sameTypeForJni<T, jbyte>)
+ res = T(env->GetByteField(obj, id));
+ else if constexpr (sameTypeForJni<T, jchar>)
+ res = T(env->GetCharField(obj, id));
+ else if constexpr (sameTypeForJni<T, jshort>)
+ res = T(env->GetShortField(obj, id));
+ else if constexpr (sameTypeForJni<T, jint>)
+ res = T(env->GetIntField(obj, id));
+ else if constexpr (sameTypeForJni<T, jlong>)
+ res = T(env->GetLongField(obj, id));
+ else if constexpr (sameTypeForJni<T, jfloat>)
+ res = T(env->GetFloatField(obj, id));
+ else if constexpr (sameTypeForJni<T, jdouble>)
+ res = T(env->GetDoubleField(obj, id));
else
QtJniTypes::staticAssertTypeMismatch();
}
@@ -672,22 +662,22 @@ private:
static constexpr void getStaticFieldForType(JNIEnv *env, T &res, jclass clazz,
jfieldID id)
{
- if constexpr (std::is_same_v<T, jboolean>)
- res = env->GetStaticBooleanField(clazz, id);
- else if constexpr (likeIntegerType<T, jbyte>)
- res = T{env->GetStaticByteField(clazz, id)};
- else if constexpr (likeIntegerType<T, jchar>)
- res = T{env->GetStaticCharField(clazz, id)};
- else if constexpr (likeIntegerType<T, jshort>)
- res = T{env->GetStaticShortField(clazz, id)};
- else if constexpr (likeIntegerType<T, jint>)
- res = T{env->GetStaticIntField(clazz, id)};
- else if constexpr (likeIntegerType<T, jlong>)
- res = T{env->GetStaticLongField(clazz, id)};
- else if constexpr (std::is_same_v<T, jfloat>)
- res = env->GetStaticFloatField(clazz, id);
- else if constexpr (std::is_same_v<T, jdouble>)
- res = env->GetStaticDoubleField(clazz, id);
+ if constexpr (sameTypeForJni<T, jboolean>)
+ res = T(env->GetStaticBooleanField(clazz, id));
+ else if constexpr (sameTypeForJni<T, jbyte>)
+ res = T(env->GetStaticByteField(clazz, id));
+ else if constexpr (sameTypeForJni<T, jchar>)
+ res = T(env->GetStaticCharField(clazz, id));
+ else if constexpr (sameTypeForJni<T, jshort>)
+ res = T(env->GetStaticShortField(clazz, id));
+ else if constexpr (sameTypeForJni<T, jint>)
+ res = T(env->GetStaticIntField(clazz, id));
+ else if constexpr (sameTypeForJni<T, jlong>)
+ res = T(env->GetStaticLongField(clazz, id));
+ else if constexpr (sameTypeForJni<T, jfloat>)
+ res = T(env->GetStaticFloatField(clazz, id));
+ else if constexpr (sameTypeForJni<T, jdouble>)
+ res = T(env->GetStaticDoubleField(clazz, id));
else
QtJniTypes::staticAssertTypeMismatch();
}
@@ -697,22 +687,22 @@ private:
jfieldID id, T value)
{
LocalFrame<T> frame;
- if constexpr (std::is_same_v<T, jboolean>)
- env->SetBooleanField(obj, id, value);
- else if constexpr (likeIntegerType<T, jbyte>)
+ if constexpr (sameTypeForJni<T, jboolean>)
+ env->SetBooleanField(obj, id, static_cast<jboolean>(value));
+ else if constexpr (sameTypeForJni<T, jbyte>)
env->SetByteField(obj, id, static_cast<jbyte>(value));
- else if constexpr (likeIntegerType<T, jchar>)
+ else if constexpr (sameTypeForJni<T, jchar>)
env->SetCharField(obj, id, static_cast<jchar>(value));
- else if constexpr (likeIntegerType<T, jshort>)
+ else if constexpr (sameTypeForJni<T, jshort>)
env->SetShortField(obj, id, static_cast<jshort>(value));
- else if constexpr (likeIntegerType<T, jint>)
+ else if constexpr (sameTypeForJni<T, jint>)
env->SetIntField(obj, id, static_cast<jint>(value));
- else if constexpr (likeIntegerType<T, jlong>)
+ else if constexpr (sameTypeForJni<T, jlong>)
env->SetLongField(obj, id, static_cast<jlong>(value));
- else if constexpr (std::is_same_v<T, jfloat>)
- env->SetFloatField(obj, id, value);
- else if constexpr (std::is_same_v<T, jdouble>)
- env->SetDoubleField(obj, id, value);
+ else if constexpr (sameTypeForJni<T, jfloat>)
+ env->SetFloatField(obj, id, static_cast<jfloat>(value));
+ else if constexpr (sameTypeForJni<T, jdouble>)
+ env->SetDoubleField(obj, id, static_cast<jdouble>(value));
else if constexpr (QtJniTypes::isObjectType<T>())
env->SetObjectField(obj, id, static_cast<jobject>(frame.convertToJni(value)));
else
@@ -724,22 +714,22 @@ private:
jfieldID id, T value)
{
LocalFrame<T> frame;
- if constexpr (std::is_same_v<T, jboolean>)
- env->SetStaticBooleanField(clazz, id, value);
- else if constexpr (likeIntegerType<T, jbyte>)
+ if constexpr (sameTypeForJni<T, jboolean>)
+ env->SetStaticBooleanField(clazz, id, static_cast<jboolean>(value));
+ else if constexpr (sameTypeForJni<T, jbyte>)
env->SetStaticByteField(clazz, id, static_cast<jbyte>(value));
- else if constexpr (likeIntegerType<T, jchar>)
+ else if constexpr (sameTypeForJni<T, jchar>)
env->SetStaticCharField(clazz, id, static_cast<jchar>(value));
- else if constexpr (likeIntegerType<T, jshort>)
+ else if constexpr (sameTypeForJni<T, jshort>)
env->SetStaticShortField(clazz, id, static_cast<jshort>(value));
- else if constexpr (likeIntegerType<T, jint>)
+ else if constexpr (sameTypeForJni<T, jint>)
env->SetStaticIntField(clazz, id, static_cast<jint>(value));
- else if constexpr (likeIntegerType<T, jlong>)
+ else if constexpr (sameTypeForJni<T, jlong>)
env->SetStaticLongField(clazz, id, static_cast<jlong>(value));
- else if constexpr (std::is_same_v<T, jfloat>)
- env->SetStaticFloatField(clazz, id, value);
- else if constexpr (std::is_same_v<T, jdouble>)
- env->SetStaticDoubleField(clazz, id, value);
+ else if constexpr (sameTypeForJni<T, jfloat>)
+ env->SetStaticFloatField(clazz, id, static_cast<jfloat>(value));
+ else if constexpr (sameTypeForJni<T, jdouble>)
+ env->SetStaticDoubleField(clazz, id, static_cast<jdouble>(value));
else if constexpr (QtJniTypes::isObjectType<T>())
env->SetStaticObjectField(clazz, id, static_cast<jobject>(frame.convertToJni(value)));
else
diff --git a/tests/auto/corelib/kernel/qjniobject/tst_qjniobject.cpp b/tests/auto/corelib/kernel/qjniobject/tst_qjniobject.cpp
index 751e5a84d8..364b0c5d62 100644
--- a/tests/auto/corelib/kernel/qjniobject/tst_qjniobject.cpp
+++ b/tests/auto/corelib/kernel/qjniobject/tst_qjniobject.cpp
@@ -77,6 +77,7 @@ private slots:
void getStaticIntField();
void getStaticByteFieldClassName();
void getStaticByteField();
+ void getStaticBooleanField();
void getStaticLongFieldClassName();
void getStaticLongField();
void getStaticDoubleFieldClassName();
@@ -936,6 +937,12 @@ void tst_QJniObject::getStaticByteField()
QCOMPARE(e, Enum::MAX_VALUE);
}
+void tst_QJniObject::getStaticBooleanField()
+{
+ QCOMPARE(TestClass::getStaticField<jboolean>("S_BOOLEAN_VAR"),
+ TestClass::getStaticField<bool>("S_BOOLEAN_VAR"));
+}
+
void tst_QJniObject::getStaticLongFieldClassName()
{
jlong i = QJniObject::getStaticField<jlong>("java/lang/Long", "MAX_VALUE");
@@ -1038,6 +1045,7 @@ void tst_QJniObject::getBooleanField()
QVERIFY(obj.isValid());
QVERIFY(obj.getField<jboolean>("BOOL_FIELD"));
+ QVERIFY(obj.getField<bool>("BOOL_FIELD"));
}
void tst_QJniObject::getIntField()
@@ -1109,6 +1117,7 @@ void tst_QJniObject::setCharField()
void tst_QJniObject::setBooleanField()
{
setField("BOOLEAN_VAR", jboolean(true));
+ setField("BOOLEAN_VAR", true);
}
void tst_QJniObject::setObjectField()
@@ -1191,6 +1200,7 @@ void tst_QJniObject::setStaticCharField()
void tst_QJniObject::setStaticBooleanField()
{
setStaticField("S_BOOLEAN_VAR", jboolean(true));
+ setStaticField("S_BOOLEAN_VAR", true);
}
void tst_QJniObject::setStaticObjectField()