blob: 1f5b7c0bad9177827ce963788315275c80939926 [file] [log] [blame]
# Copyright 2022 The Android Open Source Project
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
"""Ensure that FoV reduction with Preview Stabilization is within spec."""
import logging
import math
import os
from mobly import test_runner
import camera_properties_utils
import image_fov_utils
import image_processing_utils
import its_base_test
import its_session_utils
import opencv_processing_utils
import video_processing_utils
_PREVIEW_STABILIZATION_MODE_PREVIEW = 2
_VIDEO_DURATION = 3 # seconds
_MAX_STABILIZED_RADIUS_RATIO = 1.2 # radius of circle in stabilized preview
# should be at most 20% larger
_ROUNDESS_DELTA_THRESHOLD = 0.05
_MAX_CENTER_THRESHOLD_PERCENT = 0.075
_MAX_DIMENSION_SIZE = (1920, 1440) # max mandatory preview stream resolution
_MIN_CENTER_THRESHOLD_PERCENT = 0.02
_MIN_DIMENSION_SIZE = (176, 144) # assume QCIF to be min preview size
def _collect_data(cam, video_size, stabilize):
"""Capture a preview video from the device.
Captures camera preview frames from the passed device.
Args:
cam: camera object
video_size: str; video resolution. ex. '1920x1080'
stabilize: boolean; whether the preview should be stabilized or not
Returns:
recording object as described by cam.do_preview_recording
"""
recording_obj = cam.do_preview_recording(video_size, _VIDEO_DURATION,
stabilize)
logging.debug('Recorded output path: %s', recording_obj['recordedOutputPath'])
logging.debug('Tested quality: %s', recording_obj['quality'])
return recording_obj
def _point_distance(p1_x, p1_y, p2_x, p2_y):
"""Calculates the euclidean distance between two points.
Args:
p1_x: x coordinate of the first point
p1_y: y coordinate of the first point
p2_x: x coordinate of the second point
p2_y: y coordinate of the second point
Returns:
Euclidean distance between two points
"""
return math.sqrt(pow(p1_x - p2_x, 2) + pow(p1_y - p2_y, 2))
def _calculate_center_offset_threshold(image_size):
"""Calculates appropriate center offset threshold.
This function calculates a viable threshold that centers of two circles can be
offset by for a given image size. The threshold percent is linearly
interpolated between _MIN_CENTER_THRESHOLD_PERCENT and
_MAX_CENTER_THRESHOLD_PERCENT according to the image size passed.
Args:
image_size: pair; size of the image for which threshold has to be
calculated. ex. (1920, 1080)
Returns:
threshold value ratio between which the circle centers can differ
"""
max_diagonal = _point_distance(0, 0,
_MAX_DIMENSION_SIZE[0], _MAX_DIMENSION_SIZE[1])
min_diagonal = _point_distance(0, 0,
_MIN_DIMENSION_SIZE[0], _MIN_DIMENSION_SIZE[1])
img_diagonal = _point_distance(0, 0, image_size[0], image_size[1])
normalized_diagonal = ((img_diagonal - min_diagonal) /
(max_diagonal - min_diagonal))
if normalized_diagonal > 1 or normalized_diagonal < 0:
raise AssertionError(f'normalized diagonal > 1 or < 0!'
f' img_diag: {img_diagonal}, '
f' normalized_diagonal: {normalized_diagonal}')
# Threshold should be larger for images with smaller resolution
normalized_threshold_percent = ((1 - normalized_diagonal) *
(_MAX_CENTER_THRESHOLD_PERCENT -
_MIN_CENTER_THRESHOLD_PERCENT))
return (normalized_threshold_percent + _MIN_CENTER_THRESHOLD_PERCENT)
class PreviewStabilizationFoVTest(its_base_test.ItsBaseTest):
"""Tests if stabilized preview FoV is within spec.
The test captures two videos, one with preview stabilization on, and another
with preview stabilization off. A representative frame is selected from each
video, and analyzed to ensure that the FoV changes in the two videos are
within spec.
Specifically, the test checks for the following parameters with and without
preview stabilization:
- The circle roundness remains about constant
- The center of the circle remains relatively stable
- The size of circle changes no more that 20% i.e. the FOV changes at most
20%
"""
def test_fov_with_preview_stabilization(self):
log_path = self.log_path
with its_session_utils.ItsSession(
device_id=self.dut.serial,
camera_id=self.camera_id,
hidden_physical_id=self.hidden_physical_id) as cam:
props = cam.get_camera_properties()
props = cam.override_with_hidden_physical_camera_props(props)
# Load scene.
its_session_utils.load_scene(cam, props, self.scene,
self.tablet, chart_distance=0)
# Check skip condition
first_api_level = its_session_utils.get_first_api_level(self.dut.serial)
camera_properties_utils.skip_unless(
first_api_level >= its_session_utils.ANDROID13_API_LEVEL,
'First API level should be {} or higher. Found {}.'.format(
its_session_utils.ANDROID13_API_LEVEL, first_api_level))
supported_stabilization_modes = props[
'android.control.availableVideoStabilizationModes'
]
camera_properties_utils.skip_unless(
supported_stabilization_modes is not None
and _PREVIEW_STABILIZATION_MODE_PREVIEW
in supported_stabilization_modes,
'Preview Stabilization not supported',
)
# Raise error if not FRONT or REAR facing camera
facing = props['android.lens.facing']
if (facing != camera_properties_utils.LENS_FACING_BACK
and facing != camera_properties_utils.LENS_FACING_FRONT):
raise AssertionError('Unknown lens facing: {facing}.')
# List of video resolutions to test
supported_preview_sizes = cam.get_supported_preview_sizes(self.camera_id)
logging.debug('Supported preview resolutions: %s',
supported_preview_sizes)
test_failures = []
for video_size in supported_preview_sizes:
# recording with stabilization off
ustab_rec_obj = _collect_data(cam, video_size, False)
# recording with stabilization on
stab_rec_obj = _collect_data(cam, video_size, True)
# Grab the unstabilized video from DUT
self.dut.adb.pull([ustab_rec_obj['recordedOutputPath'], log_path])
ustab_file_name = (ustab_rec_obj['recordedOutputPath'].split('/')[-1])
logging.debug('ustab_file_name: %s', ustab_file_name)
# Grab the stabilized video from DUT
self.dut.adb.pull([stab_rec_obj['recordedOutputPath'], log_path])
stab_file_name = (stab_rec_obj['recordedOutputPath'].split('/')[-1])
logging.debug('stab_file_name: %s', stab_file_name)
# Get all frames from the videos
ustab_file_list = video_processing_utils.extract_key_frames_from_video(
log_path, ustab_file_name)
logging.debug('Number of unstabilized iframes %d', len(ustab_file_list))
stab_file_list = video_processing_utils.extract_key_frames_from_video(
log_path, stab_file_name)
logging.debug('Number of stabilized iframes %d', len(stab_file_list))
# Extract last key frame to test from each video
ustab_frame = os.path.join(log_path,
video_processing_utils
.get_key_frame_to_process(ustab_file_list))
logging.debug('unstabilized frame: %s', ustab_frame)
stab_frame = os.path.join(log_path,
video_processing_utils
.get_key_frame_to_process(stab_file_list))
logging.debug('stabilized frame: %s', stab_frame)
# Convert to numpy matrix for analysis
ustab_np_image = image_processing_utils.convert_image_to_numpy_array(
ustab_frame)
logging.debug('unstabilized frame size: %s', ustab_np_image.shape)
stab_np_image = image_processing_utils.convert_image_to_numpy_array(
stab_frame)
logging.debug('stabilized frame size: %s', stab_np_image.shape)
image_size = stab_np_image.shape
# Get circles to compare
ustab_circle = opencv_processing_utils.find_circle(
ustab_np_image,
ustab_frame,
image_fov_utils.CIRCLE_MIN_AREA,
image_fov_utils.CIRCLE_COLOR)
stab_circle = opencv_processing_utils.find_circle(
stab_np_image,
stab_frame,
image_fov_utils.CIRCLE_MIN_AREA,
image_fov_utils.CIRCLE_COLOR)
failure_string = ''
# Ensure the circles are equally round w/ and w/o stabilization
ustab_roundness = ustab_circle['w'] / ustab_circle['h']
logging.debug('unstabilized roundess: %f', ustab_roundness)
stab_roundness = stab_circle['w'] / stab_circle['h']
logging.debug('stabilized roundess: %f', stab_roundness)
roundness_diff = abs(stab_roundness - ustab_roundness)
if roundness_diff > _ROUNDESS_DELTA_THRESHOLD:
failure_string += (f'Circle roundness changed too much: '
f'unstabilized ratio: {ustab_roundness}, '
f'stabilized ratio: {stab_roundness}, '
f'Expected ratio difference <= '
f'{_ROUNDESS_DELTA_THRESHOLD}, '
f'actual ratio difference: {roundness_diff}. ')
# Distance between centers
unstab_center = (ustab_circle['x_offset'], ustab_circle['y_offset'])
logging.debug('unstabilized center: %s', unstab_center)
stab_center = (stab_circle['x_offset'], stab_circle['y_offset'])
logging.debug('stabilized center: %s', stab_center)
dist_centers = _point_distance(unstab_center[0], unstab_center[1],
stab_center[0], stab_center[1])
center_offset_threshold = _calculate_center_offset_threshold(image_size)
if dist_centers > center_offset_threshold:
failure_string += (f'Circle moved too much: '
f'unstabilized center: ('
f'{unstab_center[0]}, {unstab_center[1]}), '
f'stabilized center: '
f'({stab_center[0]}, {stab_center[1]}), '
f'expected distance < {center_offset_threshold}, '
f'actual_distance {dist_centers}. ')
# ensure radius of stabilized frame is within 120% of radius within
# unstabilized frame
ustab_radius = ustab_circle['r']
logging.debug('unstabilized radius: %f', ustab_radius)
stab_radius = stab_circle['r']
logging.debug('stabilized radius: %f', stab_radius)
max_stab_radius = ustab_radius * _MAX_STABILIZED_RADIUS_RATIO
if stab_radius > max_stab_radius:
failure_string += (f'Too much FoV reduction: '
f'unstabilized radius: {ustab_radius}, '
f'stabilized radius: {stab_radius}, '
f'expected max stabilized radius: '
f'{max_stab_radius}. ')
if failure_string:
failure_string = f'{video_size} fails FoV test. ' + failure_string
test_failures.append(failure_string)
if test_failures:
raise AssertionError(test_failures)
if __name__ == '__main__':
test_runner.main()