AP_HAL_Linux: add a shrink by software option in OpticalFlow

This shrink must be used when the output camera sizes doesn't fit
the expected output.

We don't need to crop it even when the camera sizes aren't squared since
the shrink_8bpp() function shrinks a selected area.
This commit is contained in:
Ricardo de Almeida Gonzaga 2015-05-11 01:01:54 +00:00 committed by Lucas De Marchi
parent 553d36c860
commit 753f7d864a
2 changed files with 78 additions and 29 deletions

View File

@ -117,22 +117,42 @@ void OpticalFlow_Onboard::init(AP_HAL::OpticalFlow::Gyro_Cb get_gyro)
AP_HAL::panic("OpticalFlow_Onboard: format not supported\n"); AP_HAL::panic("OpticalFlow_Onboard: format not supported\n");
} }
if (_videoin->set_crop(left, top, crop_width, crop_height)) { if (_width == HAL_OPTFLOW_ONBOARD_OUTPUT_WIDTH &&
_crop_by_software = false; _height == HAL_OPTFLOW_ONBOARD_OUTPUT_HEIGHT) {
_shrink_by_software = false;
} else { } else {
/* here we store the actual camera output width and height to use them /* here we store the actual camera output width and height to use
* later on to software crop each frame. */ * them later on to software shrink each frame. */
_crop_by_software = true; _shrink_by_software = true;
_crop_by_software_width = _width; _camera_output_width = _width;
_crop_by_software_height = _height; _camera_output_height = _height;
/* we set these values here in order to the calculations be correct /* we set these values here in order to the calculations be correct
* (such as PX4 init) even though we crop each frame later on. */ * (such as PX4 init) even though we shrink each frame later on. */
_width = HAL_OPTFLOW_ONBOARD_OUTPUT_WIDTH; _width = HAL_OPTFLOW_ONBOARD_OUTPUT_WIDTH;
_height = HAL_OPTFLOW_ONBOARD_OUTPUT_HEIGHT; _height = HAL_OPTFLOW_ONBOARD_OUTPUT_HEIGHT;
_bytesperline = HAL_OPTFLOW_ONBOARD_OUTPUT_WIDTH; _bytesperline = HAL_OPTFLOW_ONBOARD_OUTPUT_WIDTH;
} }
if (_videoin->set_crop(left, top, crop_width, crop_height)) {
_crop_by_software = false;
} else {
_crop_by_software = true;
if (!_shrink_by_software) {
/* here we store the actual camera output width and height to use
* them later on to software crop each frame. */
_camera_output_width = _width;
_camera_output_height = _height;
/* we set these values here in order to the calculations be correct
* (such as PX4 init) even though we crop each frame later on. */
_width = HAL_OPTFLOW_ONBOARD_OUTPUT_WIDTH;
_height = HAL_OPTFLOW_ONBOARD_OUTPUT_HEIGHT;
_bytesperline = HAL_OPTFLOW_ONBOARD_OUTPUT_WIDTH;
}
}
if (!_videoin->allocate_buffers(nbufs)) { if (!_videoin->allocate_buffers(nbufs)) {
AP_HAL::panic("OpticalFlow_Onboard: couldn't allocate video buffers"); AP_HAL::panic("OpticalFlow_Onboard: couldn't allocate video buffers");
} }
@ -208,14 +228,16 @@ void OpticalFlow_Onboard::_run_optflow()
Vector3f gyro_rate; Vector3f gyro_rate;
Vector2f flow_rate; Vector2f flow_rate;
VideoIn::Frame video_frame; VideoIn::Frame video_frame;
uint32_t convert_buffer_size = 0, crop_buffer_size = 0; uint32_t convert_buffer_size = 0, output_buffer_size = 0;
uint32_t crop_left = 0, crop_top = 0; uint32_t crop_left = 0, crop_top = 0;
uint8_t *convert_buffer = NULL, *crop_buffer = NULL; uint32_t shrink_scale = 0, shrink_width = 0, shrink_height = 0;
uint32_t shrink_width_offset = 0, shrink_height_offset = 0;
uint8_t *convert_buffer = NULL, *output_buffer = NULL;
uint8_t qual; uint8_t qual;
if (_format == V4L2_PIX_FMT_YUYV) { if (_format == V4L2_PIX_FMT_YUYV) {
if (_crop_by_software) { if (_shrink_by_software || _crop_by_software) {
convert_buffer_size = _crop_by_software_width * _crop_by_software_height; convert_buffer_size = _camera_output_width * _camera_output_height;
} else { } else {
convert_buffer_size = _width * _height; convert_buffer_size = _width * _height;
} }
@ -226,22 +248,38 @@ void OpticalFlow_Onboard::_run_optflow()
} }
} }
if (_crop_by_software) { if (_shrink_by_software || _crop_by_software) {
crop_buffer_size = HAL_OPTFLOW_ONBOARD_OUTPUT_WIDTH * output_buffer_size = HAL_OPTFLOW_ONBOARD_OUTPUT_WIDTH *
HAL_OPTFLOW_ONBOARD_OUTPUT_HEIGHT; HAL_OPTFLOW_ONBOARD_OUTPUT_HEIGHT;
crop_buffer = (uint8_t *)malloc(crop_buffer_size); output_buffer = (uint8_t *)malloc(output_buffer_size);
if (!crop_buffer) { if (!output_buffer) {
if (convert_buffer) { if (convert_buffer) {
free(convert_buffer); free(convert_buffer);
} }
AP_HAL::panic("OpticalFlow_Onboard: couldn't allocate crop buffer\n"); AP_HAL::panic("OpticalFlow_Onboard: couldn't allocate crop buffer\n");
} }
}
crop_left = _crop_by_software_width / 2 - if (_shrink_by_software) {
if (_camera_output_width > _camera_output_height) {
shrink_scale = (uint32_t) _camera_output_height /
HAL_OPTFLOW_ONBOARD_OUTPUT_HEIGHT;
} else {
shrink_scale = (uint32_t) _camera_output_width /
HAL_OPTFLOW_ONBOARD_OUTPUT_WIDTH;
}
shrink_width = HAL_OPTFLOW_ONBOARD_OUTPUT_WIDTH * shrink_scale;
shrink_height = HAL_OPTFLOW_ONBOARD_OUTPUT_HEIGHT * shrink_scale;
shrink_width_offset = (_camera_output_width - shrink_width) / 2;
shrink_height_offset = (_camera_output_height - shrink_height) / 2;
} else if (_crop_by_software) {
crop_left = _camera_output_width / 2 -
HAL_OPTFLOW_ONBOARD_OUTPUT_WIDTH / 2; HAL_OPTFLOW_ONBOARD_OUTPUT_WIDTH / 2;
crop_top = _crop_by_software_height / 2 - crop_top = _camera_output_height / 2 -
HAL_OPTFLOW_ONBOARD_OUTPUT_HEIGHT / 2; HAL_OPTFLOW_ONBOARD_OUTPUT_HEIGHT / 2;
} }
@ -252,8 +290,8 @@ void OpticalFlow_Onboard::_run_optflow()
free(convert_buffer); free(convert_buffer);
} }
if (crop_buffer) { if (output_buffer) {
free(crop_buffer); free(output_buffer);
} }
AP_HAL::panic("OpticalFlow_Onboard: couldn't get frame\n"); AP_HAL::panic("OpticalFlow_Onboard: couldn't get frame\n");
@ -267,14 +305,24 @@ void OpticalFlow_Onboard::_run_optflow()
memcpy(video_frame.data, convert_buffer, convert_buffer_size); memcpy(video_frame.data, convert_buffer, convert_buffer_size);
} }
if (_crop_by_software) { if (_shrink_by_software) {
VideoIn::crop_8bpp((uint8_t *)video_frame.data, crop_buffer, /* shrink_8bpp() will shrink a selected area using the offsets,
_crop_by_software_width, * therefore, we don't need the crop. */
VideoIn::shrink_8bpp((uint8_t *)video_frame.data, output_buffer,
_camera_output_width, _camera_output_height,
shrink_width_offset, shrink_width,
shrink_height_offset, shrink_height,
shrink_scale, shrink_scale);
memset(video_frame.data, 0, _camera_output_width * _camera_output_height);
memcpy(video_frame.data, output_buffer, output_buffer_size);
} else if (_crop_by_software) {
VideoIn::crop_8bpp((uint8_t *)video_frame.data, output_buffer,
_camera_output_width,
crop_left, HAL_OPTFLOW_ONBOARD_OUTPUT_WIDTH, crop_left, HAL_OPTFLOW_ONBOARD_OUTPUT_WIDTH,
crop_top, HAL_OPTFLOW_ONBOARD_OUTPUT_HEIGHT); crop_top, HAL_OPTFLOW_ONBOARD_OUTPUT_HEIGHT);
memset(video_frame.data, 0, _crop_by_software_width * _crop_by_software_height); memset(video_frame.data, 0, _camera_output_width * _camera_output_height);
memcpy(video_frame.data, crop_buffer, crop_buffer_size); memcpy(video_frame.data, output_buffer, output_buffer_size);
} }
/* if it is at least the second frame we receive /* if it is at least the second frame we receive
@ -343,8 +391,8 @@ void OpticalFlow_Onboard::_run_optflow()
free(convert_buffer); free(convert_buffer);
} }
if (crop_buffer) { if (output_buffer) {
free(crop_buffer); free(output_buffer);
} }
} }
#endif #endif

View File

@ -42,8 +42,9 @@ private:
bool _initialized; bool _initialized;
bool _data_available; bool _data_available;
bool _crop_by_software; bool _crop_by_software;
uint32_t _crop_by_software_width; bool _shrink_by_software;
uint32_t _crop_by_software_height; uint32_t _camera_output_width;
uint32_t _camera_output_height;
uint32_t _width; uint32_t _width;
uint32_t _height; uint32_t _height;
uint32_t _format; uint32_t _format;