blob: 7fa164d48ff2f5957a078930d32ac4a19eb530ca [file] [log] [blame]
from __future__ import annotations
import pytest
from pybind11_tests import class_sh_mi_thunks as m
def test_ptrdiff_drvd_base0():
ptrdiff = m.ptrdiff_drvd_base0()
# A failure here does not (necessarily) mean that there is a bug, but that
# test_class_sh_mi_thunks is not exercising what it is supposed to.
# If this ever fails on some platforms: use pytest.skip()
# If this ever fails on all platforms: don't know, seems extremely unlikely.
assert ptrdiff != 0
@pytest.mark.parametrize(
"vec_size_fn",
[
m.vec_size_base0_raw_ptr,
m.vec_size_base0_shared_ptr,
],
)
@pytest.mark.parametrize(
"get_fn",
[
m.get_drvd_as_base0_raw_ptr,
m.get_drvd_as_base0_shared_ptr,
m.get_drvd_as_base0_unique_ptr,
],
)
def test_get_vec_size_raw_shared(get_fn, vec_size_fn):
obj = get_fn()
assert vec_size_fn(obj) == 5
@pytest.mark.parametrize(
"get_fn", [m.get_drvd_as_base0_raw_ptr, m.get_drvd_as_base0_unique_ptr]
)
def test_get_vec_size_unique(get_fn):
obj = get_fn()
assert m.vec_size_base0_unique_ptr(obj) == 5
with pytest.raises(ValueError, match="Python instance was disowned"):
m.vec_size_base0_unique_ptr(obj)
def test_get_shared_vec_size_unique():
obj = m.get_drvd_as_base0_shared_ptr()
with pytest.raises(ValueError) as exc_info:
m.vec_size_base0_unique_ptr(obj)
assert (
str(exc_info.value) == "Cannot disown external shared_ptr (load_as_unique_ptr)."
)
def test_virtual_base_not_at_offset_0():
# This test ensures that the Diamond fixture actually exercises a non-zero
# virtual-base subobject offset on our supported platforms/ABIs.
#
# If this assert ever fails on some platform/toolchain, please adjust the
# C++ fixture so the virtual base is *not* at offset 0:
# - Keep VBase non-empty.
# - Make Left and Right non-empty and asymmetrically sized and, if
# needed, nudge with a modest alignment.
# - The goal is to achieve a non-zero address delta between `Diamond*`
# and `static_cast<VBase*>(Diamond*)`.
#
# Rationale: certain smart_holder features are exercised only when the
# registered subobject address differs from the most-derived object start,
# so this check guards test efficacy across compilers.
addrs = m.diamond_addrs()
assert addrs.as_vbase - addrs.as_self != 0, (
"Diamond VBase at offset 0 on this platform; to ensure test efficacy, "
"tweak fixtures (VBase/Left/Right) to ensure non-zero subobject offset."
)
@pytest.mark.parametrize(
"make_fn",
[
m.make_diamond_as_vbase_raw_ptr, # exercises smart_holder::from_raw_ptr_take_ownership
m.make_diamond_as_vbase_shared_ptr, # exercises smart_holder_from_shared_ptr
m.make_diamond_as_vbase_unique_ptr, # exercises smart_holder_from_unique_ptr
],
)
def test_make_diamond_as_vbase(make_fn):
# Added under PR #5836
vb = make_fn()
assert vb.ping() == 7
@pytest.mark.parametrize(
"clone_fn",
[
m.Tiger.clone_raw_ptr,
m.Tiger.clone_shared_ptr,
m.Tiger.clone_unique_ptr,
],
)
def test_animal_cat_tiger(clone_fn):
# Based on Animal-Cat-Tiger reproducer under PR #5796
tiger = m.Tiger()
cloned = clone_fn(tiger)
assert isinstance(cloned, m.Tiger)