Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

RSUSB support for imu sensitivity #12857

Merged
merged 15 commits into from
Apr 30, 2024
2 changes: 1 addition & 1 deletion src/ds/d400/d400-motion.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -134,7 +134,7 @@ namespace librealsense
hid_ep->get_raw_sensor()->register_metadata(RS2_FRAME_METADATA_FRAME_TIMESTAMP, make_hid_header_parser(&hid_header::timestamp));
}
//for FW >=5.16 the scale factor changes to 1000.0 since FW sends 32bit
if (_fw_version >= firmware_version( 5, 15, 1, 224))
if (_fw_version >= firmware_version( 5, 16, 0, 0))
get_raw_motion_sensor()->set_gyro_scale_factor( 10000.0 );

}
Expand Down
64 changes: 59 additions & 5 deletions src/hid/hid-device.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -100,7 +100,7 @@ namespace librealsense
{
for(auto&& p : hid_profiles)
{
set_feature_report(DEVICE_POWER_D0, _sensor_to_id[p.sensor_name], p.frequency);
set_feature_report( DEVICE_POWER_D0, _sensor_to_id[p.sensor_name], p.frequency, p.sensitivity );
}
_configured_profiles = hid_profiles;
}
Expand Down Expand Up @@ -161,10 +161,30 @@ namespace librealsense
{
if(!_running)
return;
if(r->get_actual_length() == sizeof(REALSENSE_HID_REPORT))

if( r->get_actual_length() == _realsense_hid_report_actual_size )
{
REALSENSE_HID_REPORT report;
memcpy(&report, r->get_buffer().data(), r->get_actual_length());
if( _realsense_hid_report_actual_size != sizeof( REALSENSE_HID_REPORT ) )
{
//for FW version < 5.16 the actual struct is 32 bit, so we can not use memcpy for the whole struct
//x,y,x are all short with the x variable located at offset 10, y at offset 12, and z at offset 14 within the structure.
memcpy( &report, r->get_buffer().data(), 10 );
const int16_t * x
= reinterpret_cast< const int16_t * >( r->get_buffer().data() + 10 );
const int16_t * y
= reinterpret_cast< const int16_t * >( r->get_buffer().data() + 12 );
const int16_t * z
= reinterpret_cast< const int16_t * >( r->get_buffer().data() + 14 );
report.x = *x;
report.y = *y;
report.z = *z;
memcpy( &report + 22, r->get_buffer().data() + 16, 16 );
}
else
{
memcpy( &report, r->get_buffer().data(), r->get_actual_length() );
}
_queue.enqueue(std::move(report));
}
auto sts = _messenger->submit_request(r);
Expand Down Expand Up @@ -195,7 +215,27 @@ namespace librealsense
{
REALSENSE_HID_REPORT report;
#ifdef __APPLE__
hid_read(_hidapi_device, reinterpret_cast<unsigned char*>(&report), sizeof(REALSENSE_HID_REPORT));
unsigned char tmp_buffer[100] = { 0 };
hid_read( _hidapi_device, tmp_buffer, _realsense_hid_report_actual_size );
if( _realsense_hid_report_actual_size != sizeof( REALSENSE_HID_REPORT ) )
noacoohen marked this conversation as resolved.
Show resolved Hide resolved
{
// for FW version<5.16 the actual struct is 32 bit, so we can not use memcpy for the whole struct
// x,y,x are all short with the x variable located at offset 10, y at offset 12, and
// z at offset 14 within the structure.
memcpy( &report, tmp_buffer, 10 );
Nir-Az marked this conversation as resolved.
Show resolved Hide resolved
const int16_t * x = reinterpret_cast< const int16_t * >( tmp_buffer + 10 );
const int16_t * y = reinterpret_cast< const int16_t * >( tmp_buffer + 12 );
const int16_t * z = reinterpret_cast< const int16_t * >( tmp_buffer + 14 );
report.x = *x;
Nir-Az marked this conversation as resolved.
Show resolved Hide resolved
report.y = *y;
report.z = *z;
memcpy( &report + 22, tmp_buffer + 16, 16 );
Nir-Az marked this conversation as resolved.
Show resolved Hide resolved
}
else
{
memcpy( &report, tmp_buffer, sizeof( REALSENSE_HID_REPORT ) );
}

sensor_data data{};
data.sensor = { _id_to_sensor[report.reportId] };

Expand Down Expand Up @@ -238,7 +278,7 @@ namespace librealsense
#endif
}

usb_status rs_hid_device::set_feature_report(unsigned char power, int report_id, int fps)
usb_status rs_hid_device::set_feature_report( unsigned char power, int report_id, int fps, double sensitivity)
{
uint32_t transferred;

Expand Down Expand Up @@ -273,6 +313,12 @@ namespace librealsense
if(fps > 0)
featureReport.report = (1000 / fps);

//we want to change the sensitivity values only in gyro, for FW version>=5.16
if( (int)featureReport.reportId == REPORT_ID_GYROMETER_3D
noacoohen marked this conversation as resolved.
Show resolved Hide resolved
&& _realsense_hid_report_actual_size == sizeof( REALSENSE_HID_REPORT ) )
featureReport.sensitivity = sensitivity;


res = dev->control_transfer(USB_REQUEST_CODE_SET,
HID_REQUEST_SET_REPORT,
value,
Expand Down Expand Up @@ -384,5 +430,13 @@ namespace librealsense

return ep;
}

void rs_hid_device::set_gyro_scale_factor(double scale_factor)
OhadMeir marked this conversation as resolved.
Show resolved Hide resolved
{
_gyro_scale_factor = scale_factor;
// for FW >=5.16 the scale factor changes to 1000.0 since FW sends 32bit
noacoohen marked this conversation as resolved.
Show resolved Hide resolved
if( scale_factor == 10000.0 )
Nir-Az marked this conversation as resolved.
Show resolved Hide resolved
_realsense_hid_report_actual_size = 38;
Nir-Az marked this conversation as resolved.
Show resolved Hide resolved
}
}
}
6 changes: 4 additions & 2 deletions src/hid/hid-device.h
Original file line number Diff line number Diff line change
Expand Up @@ -43,13 +43,13 @@ namespace librealsense
virtual std::vector<uint8_t> get_custom_report_data(const std::string& custom_sensor_name,
const std::string& report_name,
custom_sensor_report_field report_field) override { return {}; }
void set_gyro_scale_factor( double scale_factor ) override{};
void set_gyro_scale_factor( double scale_factor ) override;

private:
void handle_interrupt();
rs_usb_endpoint get_hid_endpoint();
rs_usb_interface get_hid_interface();
usb_status set_feature_report(unsigned char power, int report_id, int fps = 0);
usb_status set_feature_report( unsigned char power, int report_id, int fps = 0, double sensitivity = 1 );
noacoohen marked this conversation as resolved.
Show resolved Hide resolved
#ifdef __APPLE__
int hidapi_PowerDevice(unsigned char reportId);
#endif
Expand All @@ -73,6 +73,8 @@ namespace librealsense
std::vector<hid_profile> _configured_profiles;
single_consumer_queue<REALSENSE_HID_REPORT> _queue;
std::shared_ptr<active_object<>> _handle_interrupts_thread;
int _realsense_hid_report_actual_size = 32; // for FW version >=5.16 the struct changed to 38 bit
Nir-Az marked this conversation as resolved.
Show resolved Hide resolved
double _gyro_scale_factor = 10.0;
};
}
}
12 changes: 5 additions & 7 deletions src/hid/hid-types.h
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,6 @@
#define REPORT_ID_GYROMETER_3D 2
#define REPORT_ID_CUSTOM 3

const size_t SIZE_OF_HID_IMU_FRAME = 32;

static std::string gyro = "gyro_3d";
static std::string accel = "accel_3d";
Expand Down Expand Up @@ -50,16 +49,16 @@ namespace librealsense
unsigned char power;
unsigned char minReport;
unsigned short report;
unsigned short unknown;
unsigned short sensitivity;
};

struct REALSENSE_HID_REPORT {
unsigned char reportId;
unsigned char unknown;
unsigned long long timeStamp;
short x;
short y;
short z;
int32_t x;
int32_t y;
int32_t z;
unsigned int customValue1;
unsigned int customValue2;
unsigned short customValue3;
Expand All @@ -70,7 +69,6 @@ namespace librealsense
};
#pragma pack(pop)

static_assert(sizeof(REALSENSE_HID_REPORT) == SIZE_OF_HID_IMU_FRAME, "HID IMU Frame struct expected size is 32 bytes");

#pragma pack(push, 1)
struct FEATURE_REPORT
Expand All @@ -81,7 +79,7 @@ namespace librealsense
unsigned char power;
unsigned char minReport;
unsigned short report;
unsigned short unknown;
unsigned short sensitivity;
};
#pragma pack(pop)
}
Expand Down
Loading