Skip to content

Commit

Permalink
Count power state requests in device, not sensor
Browse files Browse the repository at this point in the history
  • Loading branch information
OhadMeir committed Dec 25, 2024
1 parent cf7ed8c commit 89b3009
Show file tree
Hide file tree
Showing 8 changed files with 151 additions and 70 deletions.
39 changes: 34 additions & 5 deletions src/linux/backend-v4l2.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -1814,15 +1814,44 @@ namespace librealsense

void v4l_uvc_device::set_power_state(power_state state)
{
if (state == D0 && _state == D3)
std::lock_guard< std::recursive_mutex > lock( _power_lock );
switch (state)
{
map_device_descriptor();
case D0:
{
if( _power_counter.fetch_add( 1 ) == 0 )
{
try
{
map_device_descriptor();
}
//In case of failure need to decrease use counter
catch( std::exception const & e )
{
_power_counter.fetch_add( -1 );
throw e;
}
catch( ... )
{
_power_counter.fetch_add( -1 );
throw;
}
}
break;
}
if (state == D3 && _state == D0)
case D3:
{
close(_profile);
unmap_device_descriptor();
if( _power_counter.fetch_add( -1 ) == 1 )
{
close(_profile);
unmap_device_descriptor();
}
break;
}
default:
throw std::runtime_error("illegal power state request");
}

_state = state;
}

Expand Down
2 changes: 2 additions & 0 deletions src/linux/backend-v4l2.h
Original file line number Diff line number Diff line change
Expand Up @@ -407,6 +407,8 @@ namespace librealsense

static bool get_devname_from_video_path(const std::string& real_path, std::string& devname);

std::recursive_mutex _power_lock;
std::atomic< int > _power_counter = 0;
power_state _state = D3;
std::string _name = "";
std::string _device_path = "";
Expand Down
64 changes: 51 additions & 13 deletions src/mf/mf-uvc.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -773,23 +773,54 @@ namespace librealsense
void wmf_uvc_device::set_power_state(power_state state)
{
if (state == _power_state)
return;
std::lock_guard< std::recursive_mutex > lock( _source_lock );
switch (state)
{
case D0: set_d0(); break;
case D3: set_d3(); break;
case D0:
{
if( _power_counter.fetch_add( 1 ) == 0 )
{
try
{
set_d0();
}
//In case of failure need to decrease use counter
catch( std::exception const & e )
{
_power_counter.fetch_add( -1 );
throw e;
}
catch( ... )
{
_power_counter.fetch_add( -1 );
throw;
}
}
break;
}
case D3:
{
if( _power_counter.fetch_add( -1 ) == 1 )
{
set_d3();
}
break;
}
default:
throw std::runtime_error("illegal power state request");
}
}
wmf_uvc_device::wmf_uvc_device(const uvc_device_info& info,
std::shared_ptr<const wmf_backend> backend)
: _streamIndex(MAX_PINS), _info(info), _is_flushed(), _has_started(), _backend(std::move(backend)),
_systemwide_lock(info.unique_id.c_str(), WAIT_FOR_MUTEX_TIME_OUT),
_location(""), _device_usb_spec(usb3_type)
wmf_uvc_device::wmf_uvc_device( const uvc_device_info & info, std::shared_ptr< const wmf_backend > backend )
: _streamIndex( MAX_PINS )
, _info( info )
, _is_flushed()
, _has_started()
, _backend( std::move( backend ) )
, _systemwide_lock( info.unique_id.c_str(), WAIT_FOR_MUTEX_TIME_OUT )
, _location( "" )
, _device_usb_spec( usb3_type )
{
if (!is_connected(info))
{
Expand Down Expand Up @@ -884,7 +915,6 @@ namespace librealsense
//enable reader
CHECK_HR(MFCreateSourceReaderFromMediaSource(_source, _reader_attrs, &_reader));
CHECK_HR(_reader->SetStreamSelection(static_cast<DWORD>(MF_SOURCE_READER_ALL_STREAMS), TRUE));
_power_state = D0;
for( auto && xu : _xus )
init_xu( xu );
Expand All @@ -899,7 +929,6 @@ namespace librealsense
safe_release(_source);
for (auto& elem : _streams)
elem.callback = nullptr;
_power_state = D3;
}
void wmf_uvc_device::foreach_profile(std::function<void(const mf_profile& profile, CComPtr<IMFMediaType> media_type, bool& quit)> action) const
Expand Down Expand Up @@ -1210,5 +1239,14 @@ namespace librealsense
_profiles.clear();
_frame_callbacks.clear();
}
}
}

power_state wmf_uvc_device::get_power_state() const
{
LOG_ERROR( "wmf_uvc_device::get_power_state start. this = " << this );
std::lock_guard< std::recursive_mutex > lock( _source_lock );
std::string tmp = _source ? "D0" : "D3";
LOG_ERROR( "wmf_uvc_device::get_power_state got lock. power state is " << tmp );
return _source ? D0 : D3;
}
} //namespace platform
} //namespace librealsense
5 changes: 3 additions & 2 deletions src/mf/mf-uvc.h
Original file line number Diff line number Diff line change
Expand Up @@ -73,7 +73,7 @@ namespace librealsense
void stop_callbacks() override;
void close(stream_profile profile) override;
void set_power_state(power_state state) override;
power_state get_power_state() const override { return _power_state; }
power_state get_power_state() const override;
std::vector<stream_profile> get_profiles() const override;

static bool is_connected(const uvc_device_info& info);
Expand Down Expand Up @@ -119,7 +119,6 @@ namespace librealsense
std::shared_ptr<const wmf_backend> _backend;

const uvc_device_info _info;
power_state _power_state = D3;

CComPtr<IMFSourceReader> _reader = nullptr;
CComPtr<IMFMediaSource> _source = nullptr;
Expand All @@ -138,6 +137,7 @@ namespace librealsense
std::vector<profile_and_callback> _streams;
std::mutex _streams_mutex;

mutable std::recursive_mutex _source_lock; // Guarding access to _source
named_mutex _systemwide_lock;
std::string _location;
usb_spec _device_usb_spec;
Expand All @@ -147,6 +147,7 @@ namespace librealsense
bool _streaming = false;
std::atomic<bool> _is_started = false;
std::wstring _device_id;
std::atomic< int > _power_counter = 0;
std::vector< platform::extension_unit > _xus;
};

Expand Down
57 changes: 22 additions & 35 deletions src/uvc-sensor.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -31,7 +31,6 @@ uvc_sensor::uvc_sensor( std::string const & name,
device * dev )
: super( name, dev )
, _device( std::move( uvc_device ) )
, _user_count( 0 )
, _timestamp_reader( std::move( timestamp_reader ) )
, _gyro_counter(0)
, _accel_counter(0)
Expand Down Expand Up @@ -413,47 +412,35 @@ void uvc_sensor::reset_streaming()

void uvc_sensor::acquire_power()
{
std::lock_guard< std::mutex > lock( _power_lock );
if( _user_count.fetch_add( 1 ) == 0 )
try
{
try
{
_device->set_power_state( platform::D0 );
}
catch( std::exception const & e )
{
_user_count.fetch_add( -1 );
LOG_ERROR( "acquire_power failed: " << e.what() );
throw;
}
catch( ... )
{
_user_count.fetch_add( -1 );
LOG_ERROR( "acquire_power failed" );
throw;
}
_device->set_power_state( platform::D0 );
}
catch( std::exception const & e )
{
LOG_ERROR( "acquire_power failed: " << e.what() );
throw;
}
catch( ... )
{
LOG_ERROR( "acquire_power failed" );
throw;
}
}

void uvc_sensor::release_power()
{
std::lock_guard< std::mutex > lock( _power_lock );
if( _user_count.fetch_add( -1 ) == 1 )
try
{
try
{
_device->set_power_state( platform::D3 );
}
catch( std::exception const & e )
{
// TODO may need to change the user-count?
LOG_ERROR( "release_power failed: " << e.what() );
}
catch( ... )
{
// TODO may need to change the user-count?
LOG_ERROR( "release_power failed" );
}
_device->set_power_state( platform::D3 );
}
catch( std::exception const & e )
{
LOG_ERROR( "release_power failed: " << e.what() );
}
catch( ... )
{
LOG_ERROR( "release_power failed" );
}
}

Expand Down
2 changes: 0 additions & 2 deletions src/uvc-sensor.h
Original file line number Diff line number Diff line change
Expand Up @@ -101,8 +101,6 @@ class uvc_sensor : public raw_sensor_base

std::shared_ptr< platform::uvc_device > _device;
std::vector< platform::stream_profile > _internal_config;
std::atomic< int > _user_count;
std::mutex _power_lock;
std::mutex _configure_lock;
std::unique_ptr< power > _power;
std::unique_ptr< frame_timestamp_reader > _timestamp_reader;
Expand Down
49 changes: 37 additions & 12 deletions src/uvc/uvc-device.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -174,28 +174,53 @@ namespace librealsense
{
if(state != _power_state)
{
std::lock_guard< std::recursive_mutex > lock( _power_lock );
switch(state)
{
case D0:
_messenger = _usb_device->open(static_cast<uint8_t>(_info.mi));
if (_messenger)
if( _power_counter.fetch_add( 1 ) == 0 )
{
try{
listen_to_interrupts();
} catch(const std::exception& exception) {
// this exception catching avoids crash when disconnecting 2 devices at once - bug seen in android os
LOG_WARNING("rs_uvc_device exception in listen_to_interrupts method: " << exception.what());
try
{
_messenger = _usb_device->open(static_cast<uint8_t>(_info.mi));
if (_messenger)
{
try
{
listen_to_interrupts();
}
catch(const std::exception& exception)
{
// this exception catching avoids crash when disconnecting 2 devices at once - bug seen in android os
LOG_WARNING("rs_uvc_device exception in listen_to_interrupts method: " << exception.what());
}
_power_state = D0;
}
}
//In case of failure need to decrease use counter
catch( std::exception const & e )
{
_power_counter.fetch_add( -1 );
throw e;
}
catch( ... )
{
_power_counter.fetch_add( -1 );
throw;
}
_power_state = D0;
}

break;
case D3:
if(_messenger)
if( _power_counter.fetch_add( -1 ) == 1 )
{
close_uvc_device();
_messenger.reset();
if( _messenger )
{
close_uvc_device();
_messenger.reset();
}
_power_state = D3;
}
_power_state = D3;
break;
}
}
Expand Down
3 changes: 2 additions & 1 deletion src/uvc/uvc-device.h
Original file line number Diff line number Diff line change
Expand Up @@ -96,7 +96,8 @@ namespace librealsense

const uvc_device_info _info;
power_state _power_state = D3; // power state change is unsupported

std::recursive_mutex _power_lock;
std::atomic< int > _power_counter = 0;
std::vector<profile_and_callback> _streams;

std::string _location;
Expand Down

0 comments on commit 89b3009

Please sign in to comment.