mirror of
https://github.com/SFML/SFML.git
synced 2024-11-28 22:31:09 +08:00
Add move semantics to 'AlResource'
This commit is contained in:
parent
5b500ad2c2
commit
8a11078075
@ -31,6 +31,15 @@
|
|||||||
#include <SFML/Audio/Export.hpp>
|
#include <SFML/Audio/Export.hpp>
|
||||||
|
|
||||||
|
|
||||||
|
namespace sf::priv
|
||||||
|
{
|
||||||
|
////////////////////////////////////////////////////////////
|
||||||
|
/// \brief Return OpenAL resource count
|
||||||
|
///
|
||||||
|
////////////////////////////////////////////////////////////
|
||||||
|
SFML_AUDIO_API unsigned int getOpenAlResourceCount();
|
||||||
|
}
|
||||||
|
|
||||||
namespace sf
|
namespace sf
|
||||||
{
|
{
|
||||||
////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////
|
||||||
@ -52,6 +61,48 @@ protected:
|
|||||||
///
|
///
|
||||||
////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////
|
||||||
~AlResource();
|
~AlResource();
|
||||||
|
|
||||||
|
////////////////////////////////////////////////////////////
|
||||||
|
/// \brief Copy constructor
|
||||||
|
///
|
||||||
|
////////////////////////////////////////////////////////////
|
||||||
|
AlResource(const AlResource& other);
|
||||||
|
|
||||||
|
////////////////////////////////////////////////////////////
|
||||||
|
/// \brief Move constructor
|
||||||
|
///
|
||||||
|
////////////////////////////////////////////////////////////
|
||||||
|
AlResource(AlResource&& other) noexcept;
|
||||||
|
|
||||||
|
////////////////////////////////////////////////////////////
|
||||||
|
/// \brief Copy assignment operator
|
||||||
|
///
|
||||||
|
////////////////////////////////////////////////////////////
|
||||||
|
AlResource& operator=(const AlResource& other);
|
||||||
|
|
||||||
|
////////////////////////////////////////////////////////////
|
||||||
|
/// \brief Move assignment operator
|
||||||
|
///
|
||||||
|
////////////////////////////////////////////////////////////
|
||||||
|
AlResource& operator=(AlResource&& other) noexcept;
|
||||||
|
|
||||||
|
private:
|
||||||
|
////////////////////////////////////////////////////////////
|
||||||
|
/// \brief Increment resource counter
|
||||||
|
///
|
||||||
|
////////////////////////////////////////////////////////////
|
||||||
|
void incrementResourceCounter();
|
||||||
|
|
||||||
|
////////////////////////////////////////////////////////////
|
||||||
|
/// \brief Decrement resource counter
|
||||||
|
///
|
||||||
|
////////////////////////////////////////////////////////////
|
||||||
|
void decrementResourceCounter();
|
||||||
|
|
||||||
|
////////////////////////////////////////////////////////////
|
||||||
|
// Member data
|
||||||
|
////////////////////////////////////////////////////////////
|
||||||
|
bool m_active; //!< Whether the resource is active or not
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace sf
|
} // namespace sf
|
||||||
|
@ -29,11 +29,13 @@
|
|||||||
#include <SFML/Audio/AudioDevice.hpp>
|
#include <SFML/Audio/AudioDevice.hpp>
|
||||||
#include <memory>
|
#include <memory>
|
||||||
#include <mutex>
|
#include <mutex>
|
||||||
|
#include <utility>
|
||||||
|
#include <cassert>
|
||||||
|
|
||||||
|
|
||||||
namespace
|
namespace
|
||||||
{
|
{
|
||||||
// OpenAL resources counter and its mutex
|
// OpenAL resource counter and its mutex
|
||||||
unsigned int count = 0;
|
unsigned int count = 0;
|
||||||
std::recursive_mutex mutex;
|
std::recursive_mutex mutex;
|
||||||
|
|
||||||
@ -44,10 +46,122 @@ namespace
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
namespace sf::priv
|
||||||
|
{
|
||||||
|
////////////////////////////////////////////////////////////
|
||||||
|
/// \brief Return OpenAL resource count
|
||||||
|
///
|
||||||
|
////////////////////////////////////////////////////////////
|
||||||
|
unsigned int getOpenAlResourceCount()
|
||||||
|
{
|
||||||
|
std::scoped_lock lock(mutex);
|
||||||
|
return count;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
namespace sf
|
namespace sf
|
||||||
{
|
{
|
||||||
////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////
|
||||||
AlResource::AlResource()
|
AlResource::AlResource() :
|
||||||
|
m_active(true)
|
||||||
|
{
|
||||||
|
incrementResourceCounter();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
////////////////////////////////////////////////////////////
|
||||||
|
AlResource::~AlResource()
|
||||||
|
{
|
||||||
|
if (m_active)
|
||||||
|
decrementResourceCounter();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
////////////////////////////////////////////////////////////
|
||||||
|
AlResource::AlResource(const AlResource& other) :
|
||||||
|
m_active(other.m_active)
|
||||||
|
{
|
||||||
|
if (m_active)
|
||||||
|
incrementResourceCounter();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
////////////////////////////////////////////////////////////
|
||||||
|
AlResource::AlResource(AlResource&& other) noexcept :
|
||||||
|
m_active(std::exchange(other.m_active, false))
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
////////////////////////////////////////////////////////////
|
||||||
|
AlResource& AlResource::operator=(const AlResource& other)
|
||||||
|
{
|
||||||
|
if (&other == this)
|
||||||
|
return *this;
|
||||||
|
|
||||||
|
if (m_active && other.m_active)
|
||||||
|
{
|
||||||
|
// Both resources are already accounted for, nothing to do
|
||||||
|
}
|
||||||
|
else if (!m_active && other.m_active)
|
||||||
|
{
|
||||||
|
// '*this' was previously moved-from
|
||||||
|
m_active = true;
|
||||||
|
incrementResourceCounter();
|
||||||
|
}
|
||||||
|
else if (m_active && !other.m_active)
|
||||||
|
{
|
||||||
|
// 'other' was previously moved-from
|
||||||
|
m_active = false;
|
||||||
|
decrementResourceCounter();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// Both resources are inactive, nothing to do
|
||||||
|
assert(!m_active && !other.m_active);
|
||||||
|
}
|
||||||
|
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
////////////////////////////////////////////////////////////
|
||||||
|
AlResource& AlResource::operator=(AlResource&& other) noexcept
|
||||||
|
{
|
||||||
|
if (&other == this)
|
||||||
|
return *this;
|
||||||
|
|
||||||
|
if (m_active && other.m_active)
|
||||||
|
{
|
||||||
|
other.m_active = false;
|
||||||
|
decrementResourceCounter();
|
||||||
|
}
|
||||||
|
else if (!m_active && other.m_active)
|
||||||
|
{
|
||||||
|
// '*this' was previously moved-from
|
||||||
|
m_active = true;
|
||||||
|
other.m_active = false;
|
||||||
|
}
|
||||||
|
else if (m_active && !other.m_active)
|
||||||
|
{
|
||||||
|
// 'other' was previously moved-from
|
||||||
|
m_active = false;
|
||||||
|
decrementResourceCounter();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// Both resources are inactive, nothing to do
|
||||||
|
assert(!m_active && !other.m_active);
|
||||||
|
}
|
||||||
|
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
////////////////////////////////////////////////////////////
|
||||||
|
void AlResource::incrementResourceCounter()
|
||||||
{
|
{
|
||||||
// Protect from concurrent access
|
// Protect from concurrent access
|
||||||
std::scoped_lock lock(mutex);
|
std::scoped_lock lock(mutex);
|
||||||
@ -57,20 +171,20 @@ AlResource::AlResource()
|
|||||||
globalDevice = std::make_unique<priv::AudioDevice>();
|
globalDevice = std::make_unique<priv::AudioDevice>();
|
||||||
|
|
||||||
// Increment the resources counter
|
// Increment the resources counter
|
||||||
count++;
|
++count;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////
|
||||||
AlResource::~AlResource()
|
void AlResource::decrementResourceCounter()
|
||||||
{
|
{
|
||||||
// Protect from concurrent access
|
// Protect from concurrent access
|
||||||
std::scoped_lock lock(mutex);
|
std::scoped_lock lock(mutex);
|
||||||
|
|
||||||
// Decrement the resources counter
|
// Decrement the resources counter
|
||||||
count--;
|
--count;
|
||||||
|
|
||||||
// If there's no more resource alive, we can destroy the device
|
// If there are no more resources alive, we can destroy the device
|
||||||
if (count == 0)
|
if (count == 0)
|
||||||
globalDevice.reset();
|
globalDevice.reset();
|
||||||
}
|
}
|
||||||
|
217
test/Audio/AlResource.cpp
Normal file
217
test/Audio/AlResource.cpp
Normal file
@ -0,0 +1,217 @@
|
|||||||
|
#include <SFML/Audio/AlResource.hpp>
|
||||||
|
#include <utility>
|
||||||
|
|
||||||
|
#include <doctest.h>
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
|
||||||
|
struct TestAlResource : sf::AlResource { };
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE("sf::AlResource class - [audio]")
|
||||||
|
{
|
||||||
|
SUBCASE("Never used")
|
||||||
|
{
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 0u);
|
||||||
|
}
|
||||||
|
|
||||||
|
SUBCASE("Basic ctor/dtor")
|
||||||
|
{
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 0u);
|
||||||
|
|
||||||
|
{
|
||||||
|
TestAlResource r0;
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 1u);
|
||||||
|
}
|
||||||
|
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 0u);
|
||||||
|
}
|
||||||
|
|
||||||
|
SUBCASE("Nested ctor/dtor")
|
||||||
|
{
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 0u);
|
||||||
|
|
||||||
|
{
|
||||||
|
TestAlResource r0;
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 1u);
|
||||||
|
|
||||||
|
{
|
||||||
|
TestAlResource r1;
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 2u);
|
||||||
|
}
|
||||||
|
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 1u);
|
||||||
|
}
|
||||||
|
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 0u);
|
||||||
|
}
|
||||||
|
|
||||||
|
SUBCASE("Copy ctor")
|
||||||
|
{
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 0u);
|
||||||
|
|
||||||
|
{
|
||||||
|
TestAlResource r0;
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 1u);
|
||||||
|
|
||||||
|
{
|
||||||
|
TestAlResource r1 = r0;
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 2u);
|
||||||
|
}
|
||||||
|
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 1u);
|
||||||
|
}
|
||||||
|
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 0u);
|
||||||
|
}
|
||||||
|
|
||||||
|
SUBCASE("Move ctor")
|
||||||
|
{
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 0u);
|
||||||
|
|
||||||
|
{
|
||||||
|
TestAlResource r0;
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 1u);
|
||||||
|
|
||||||
|
{
|
||||||
|
TestAlResource r1 = std::move(r0);
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 1u);
|
||||||
|
}
|
||||||
|
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 0u);
|
||||||
|
}
|
||||||
|
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 0u);
|
||||||
|
}
|
||||||
|
|
||||||
|
SUBCASE("Copy assignment from active to active")
|
||||||
|
{
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 0u);
|
||||||
|
|
||||||
|
{
|
||||||
|
TestAlResource r0;
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 1u);
|
||||||
|
|
||||||
|
TestAlResource r1;
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 2u);
|
||||||
|
|
||||||
|
r1 = r0;
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 2u);
|
||||||
|
}
|
||||||
|
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 0u);
|
||||||
|
}
|
||||||
|
|
||||||
|
SUBCASE("Copy assignment from inactive to active")
|
||||||
|
{
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 0u);
|
||||||
|
|
||||||
|
{
|
||||||
|
TestAlResource r0;
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 1u);
|
||||||
|
|
||||||
|
TestAlResource r1 = std::move(r0);
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 1u);
|
||||||
|
|
||||||
|
r1 = r0;
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 0u);
|
||||||
|
}
|
||||||
|
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 0u);
|
||||||
|
}
|
||||||
|
|
||||||
|
SUBCASE("Move assignment from active to active")
|
||||||
|
{
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 0u);
|
||||||
|
|
||||||
|
{
|
||||||
|
TestAlResource r0;
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 1u);
|
||||||
|
|
||||||
|
TestAlResource r1;
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 2u);
|
||||||
|
|
||||||
|
r1 = std::move(r0);
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 1u);
|
||||||
|
}
|
||||||
|
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 0u);
|
||||||
|
}
|
||||||
|
|
||||||
|
SUBCASE("Move assignment from inactive to active")
|
||||||
|
{
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 0u);
|
||||||
|
|
||||||
|
{
|
||||||
|
TestAlResource r0;
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 1u);
|
||||||
|
|
||||||
|
TestAlResource r1 = std::move(r0);
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 1u);
|
||||||
|
|
||||||
|
r1 = std::move(r0);
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 0u);
|
||||||
|
}
|
||||||
|
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 0u);
|
||||||
|
}
|
||||||
|
|
||||||
|
SUBCASE("Copy assignment from active to inactive")
|
||||||
|
{
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 0u);
|
||||||
|
|
||||||
|
{
|
||||||
|
TestAlResource r0;
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 1u);
|
||||||
|
|
||||||
|
TestAlResource r1 = std::move(r0);
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 1u);
|
||||||
|
|
||||||
|
r1 = r0;
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 0u);
|
||||||
|
}
|
||||||
|
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 0u);
|
||||||
|
}
|
||||||
|
|
||||||
|
SUBCASE("Move assignment from active to inactive")
|
||||||
|
{
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 0u);
|
||||||
|
|
||||||
|
{
|
||||||
|
TestAlResource r0;
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 1u);
|
||||||
|
|
||||||
|
TestAlResource r1 = std::move(r0);
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 1u);
|
||||||
|
|
||||||
|
r1 = std::move(r0);
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 0u);
|
||||||
|
}
|
||||||
|
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 0u);
|
||||||
|
}
|
||||||
|
|
||||||
|
SUBCASE("Copy assignment from inactive to inactive")
|
||||||
|
{
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 0u);
|
||||||
|
|
||||||
|
{
|
||||||
|
TestAlResource r0;
|
||||||
|
TestAlResource r1;
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 2u);
|
||||||
|
|
||||||
|
TestAlResource r2 = std::move(r0);
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 2u);
|
||||||
|
|
||||||
|
TestAlResource r3 = std::move(r1);
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 2u);
|
||||||
|
|
||||||
|
r0 = r1;
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 2u);
|
||||||
|
}
|
||||||
|
|
||||||
|
CHECK(sf::priv::getOpenAlResourceCount() == 0u);
|
||||||
|
}
|
||||||
|
}
|
@ -53,7 +53,7 @@ endif()
|
|||||||
|
|
||||||
if(SFML_BUILD_AUDIO)
|
if(SFML_BUILD_AUDIO)
|
||||||
SET(AUDIO_SRC
|
SET(AUDIO_SRC
|
||||||
"${SRCROOT}/Audio/Dummy.cpp" # TODO: Remove when there are real tests
|
"${SRCROOT}/Audio/AlResource.cpp"
|
||||||
)
|
)
|
||||||
sfml_add_test(test-sfml-audio "${AUDIO_SRC}" SFML::Audio)
|
sfml_add_test(test-sfml-audio "${AUDIO_SRC}" SFML::Audio)
|
||||||
target_link_libraries(test-sfml-audio PRIVATE sfml-test-main)
|
target_link_libraries(test-sfml-audio PRIVATE sfml-test-main)
|
||||||
|
Loading…
Reference in New Issue
Block a user