Skip to content

Commit

Permalink
qtypes: add support for Qt alias types that don't match
Browse files Browse the repository at this point in the history
Some times don't match the Rust types so add these missing types.

Closes KDAB#882
  • Loading branch information
ahayzen-kdab committed Oct 11, 2024
1 parent e6ee353 commit 824aa44
Show file tree
Hide file tree
Showing 5 changed files with 319 additions and 0 deletions.
2 changes: 2 additions & 0 deletions crates/cxx-qt-lib/build.rs
Original file line number Diff line number Diff line change
Expand Up @@ -147,6 +147,7 @@ fn main() {
"core/qstringlist",
"core/qt",
"core/qtime",
"core/qtypes",
"core/qurl",
"core/qvariant/mod",
"core/qvariant/qvariant_bool",
Expand Down Expand Up @@ -271,6 +272,7 @@ fn main() {
"core/qstring",
"core/qstringlist",
"core/qtime",
"core/qtypes",
"core/qurl",
"core/qvariant/qvariant",
"core/qvector/qvector",
Expand Down
44 changes: 44 additions & 0 deletions crates/cxx-qt-lib/include/core/qtypes.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,44 @@
// clang-format off
// SPDX-FileCopyrightText: 2024 Klarälvdalens Datakonsult AB, a KDAB Group company <info@kdab.com>
// clang-format on
// SPDX-FileContributor: Andrew Hayzen <andrew.hayzen@kdab.com>
//
// SPDX-License-Identifier: MIT OR Apache-2.0
#pragma once

#include <cstdint>

#include <QtCore/Qt>

#include "rust/cxx.h"

namespace rust {
namespace cxxqtlib1 {

::qint64
qint64FromI64(::std::int64_t value);
::std::int64_t
qint64IntoI64(::qint64 value);

::qintptr
qintptrFromIsize(::rust::isize value);
::rust::isize
qintptrIntoIsize(qintptr value);

::quint64
quint64FromU64(::std::uint64_t value);
::std::uint64_t
quint64IntoU64(::quint64 value);

::quintptr
quintptrFromUsize(::rust::usize value);
::rust::usize
quintptrIntoUsize(quintptr value);

::qsizetype
qsizetypeFromIsize(::rust::isize value);
::rust::isize
qsizetypeIntoIsize(qsizetype value);

}
}
3 changes: 3 additions & 0 deletions crates/cxx-qt-lib/src/core/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -83,6 +83,9 @@ pub use qt::{
mod qtime;
pub use qtime::QTime;

mod qtypes;
pub use qtypes::{QInt64, QIntPtr, QSizeType, QUInt64, QUIntPtr};

#[cfg(not(target_os = "emscripten"))]
mod qtimezone;
#[cfg(not(target_os = "emscripten"))]
Expand Down
81 changes: 81 additions & 0 deletions crates/cxx-qt-lib/src/core/qtypes.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,81 @@
// clang-format off
// SPDX-FileCopyrightText: 2024 Klarälvdalens Datakonsult AB, a KDAB Group company <info@kdab.com>
// clang-format on
// SPDX-FileContributor: Andrew Hayzen <andrew.hayzen@kdab.com>
//
// SPDX-License-Identifier: MIT OR Apache-2.0
#include "cxx-qt-lib/qtypes.h"

#include "cxx-qt-lib/assertion_utils.h"

assert_alignment_and_size(qint64, { ::std::int64_t a0; });
assert_alignment_and_size(qintptr, { ::std::intptr_t a0; });
assert_alignment_and_size(quint64, { ::std::uint64_t a0; });
assert_alignment_and_size(quintptr, { ::std::uintptr_t a0; });
assert_alignment_and_size(qsizetype, { ::std::size_t a0; });

namespace rust {
namespace cxxqtlib1 {

::qint64
qint64FromI64(::std::int64_t value)
{
return static_cast<::qint64>(value);
}

::std::int64_t
qint64IntoI64(::qint64 value)
{
return static_cast<::std::int64_t>(value);
}

::qintptr
qintptrFromIsize(::rust::isize value)
{
return static_cast<::qintptr>(value);
}

::rust::isize
qintptrIntoIsize(::qintptr value)
{
return static_cast<::rust::isize>(value);
}

::quint64
quint64FromU64(::std::uint64_t value)
{
return static_cast<::quint64>(value);
}

::std::uint64_t
quint64IntoU64(::quint64 value)
{
return static_cast<::std::uint64_t>(value);
}

::quintptr
quintptrFromUsize(::rust::usize value)
{
return static_cast<::quintptr>(value);
}

::rust::usize
quintptrIntoUsize(::quintptr value)
{
return static_cast<::rust::usize>(value);
}

::qsizetype
qsizetypeFromIsize(::rust::isize value)
{
return static_cast<::qsizetype>(value);
}

::rust::isize
qsizetypeIntoIsize(::qsizetype value)
{
return static_cast<::rust::isize>(value);
}

}
}
189 changes: 189 additions & 0 deletions crates/cxx-qt-lib/src/core/qtypes.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,189 @@
// SPDX-FileCopyrightText: 2024 Klarälvdalens Datakonsult AB, a KDAB Group company <info@kdab.com>
// SPDX-FileContributor: Andrew Hayzen <andrew.hayzen@kdab.com>
//
// SPDX-License-Identifier: MIT OR Apache-2.0

use cxx::{type_id, ExternType};
use std::mem::MaybeUninit;

#[cxx::bridge]
mod ffi {
unsafe extern "C++" {
include!("cxx-qt-lib/qtypes.h");

#[cxx_name = "qint64"]
type QInt64 = super::QInt64;

#[cxx_name = "qintptr"]
type QIntPtr = super::QIntPtr;

#[cxx_name = "quint64"]
type QUInt64 = super::QUInt64;

#[cxx_name = "quintptr"]
type QUIntPtr = super::QUIntPtr;

#[cxx_name = "qsizetype"]
type QSizeType = super::QSizeType;
}

#[namespace = "rust::cxxqtlib1"]
unsafe extern "C++" {
#[rust_name = "qint64_from_i64"]
fn qint64FromI64(value: i64) -> QInt64;
#[rust_name = "qint64_into_i64"]
fn qint64IntoI64(value: QInt64) -> i64;

#[rust_name = "qintptr_from_isize"]
fn qintptrFromIsize(value: isize) -> QIntPtr;
#[rust_name = "qintptr_into_isize"]
fn qintptrIntoIsize(value: QIntPtr) -> isize;

#[rust_name = "quint64_from_u64"]
fn quint64FromU64(value: u64) -> QUInt64;
#[rust_name = "quint64_into_u64"]
fn quint64IntoU64(value: QUInt64) -> u64;

#[rust_name = "quintptr_from_usize"]
fn quintptrFromUsize(value: usize) -> QUIntPtr;
#[rust_name = "quintptr_into_usize"]
fn quintptrIntoUsize(value: QUIntPtr) -> usize;

#[rust_name = "qsizetype_from_isize"]
fn qsizetypeFromIsize(value: isize) -> QSizeType;
#[rust_name = "qsizetype_into_isize"]
fn qsizetypeIntoIsize(value: QSizeType) -> isize;
}
}

/// Typedef for long long int. This type is guaranteed to be 64-bit on all platforms supported by Qt.
#[repr(C)]
pub struct QInt64 {
_space: MaybeUninit<i64>,
}

impl From<i64> for QInt64 {
fn from(value: i64) -> Self {
ffi::qint64_from_i64(value)
}
}

impl From<QInt64> for i64 {
fn from(value: QInt64) -> Self {
ffi::qint64_into_i64(value)
}
}

// Safety:
//
// Static checks on the C++ side to ensure the size is the same.
unsafe impl ExternType for QInt64 {
type Id = type_id!("qint64");
type Kind = cxx::kind::Trivial;
}

/// Integral type for representing pointers in a signed integer (useful for hashing, etc.).
#[repr(C)]
pub struct QIntPtr {
_space: MaybeUninit<isize>,
}

impl From<isize> for QIntPtr {
fn from(value: isize) -> Self {
ffi::qintptr_from_isize(value)
}
}

impl From<QIntPtr> for isize {
fn from(value: QIntPtr) -> Self {
ffi::qintptr_into_isize(value)
}
}

// Safety:
//
// Static checks on the C++ side to ensure the size is the same.
unsafe impl ExternType for QIntPtr {
type Id = type_id!("qintptr");
type Kind = cxx::kind::Trivial;
}

/// Typedef for unsigned long long int. This type is guaranteed to be 64-bit on all platforms supported by Qt.
#[repr(C)]
pub struct QUInt64 {
_space: MaybeUninit<u64>,
}

impl From<u64> for QUInt64 {
fn from(value: u64) -> Self {
ffi::quint64_from_u64(value)
}
}

impl From<QUInt64> for u64 {
fn from(value: QUInt64) -> Self {
ffi::quint64_into_u64(value)
}
}

// Safety:
//
// Static checks on the C++ side to ensure the size is the same.
unsafe impl ExternType for QUInt64 {
type Id = type_id!("quint64");
type Kind = cxx::kind::Trivial;
}

/// Integral type for representing pointers in an unsigned integer (useful for hashing, etc.).
#[repr(C)]
pub struct QUIntPtr {
_space: MaybeUninit<usize>,
}

impl From<usize> for QUIntPtr {
fn from(value: usize) -> Self {
ffi::quintptr_from_usize(value)
}
}

impl From<QUIntPtr> for usize {
fn from(value: QUIntPtr) -> Self {
ffi::quintptr_into_usize(value)
}
}

// Safety:
//
// Static checks on the C++ side to ensure the size is the same.
unsafe impl ExternType for QUIntPtr {
type Id = type_id!("quintptr");
type Kind = cxx::kind::Trivial;
}

/// Integral type providing Posix' ssize_t for all platforms.
///
/// This type is guaranteed to be the same size as a size_t on all platforms supported by Qt.
#[repr(C)]
pub struct QSizeType {
_space: MaybeUninit<isize>,
}

impl From<isize> for QSizeType {
fn from(value: isize) -> Self {
ffi::qsizetype_from_isize(value)
}
}

impl From<QSizeType> for isize {
fn from(value: QSizeType) -> Self {
ffi::qsizetype_into_isize(value)
}
}

// Safety:
//
// Static checks on the C++ side to ensure the size is the same.
unsafe impl ExternType for QSizeType {
type Id = type_id!("qsizetype");
type Kind = cxx::kind::Trivial;
}

0 comments on commit 824aa44

Please sign in to comment.