2 * Copyright (C) 2017 The Android Open Source Project
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
8 * http://www.apache.org/licenses/LICENSE-2.0
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
17 #define LOG_TAG "VtsHalAudioV2_0TargetTest"
26 #include <type_traits>
29 #include <VtsHalHidlTargetTestBase.h>
31 #include <android-base/logging.h>
33 #include <android/hardware/audio/2.0/IDevice.h>
34 #include <android/hardware/audio/2.0/IDevicesFactory.h>
35 #include <android/hardware/audio/2.0/IPrimaryDevice.h>
36 #include <android/hardware/audio/2.0/types.h>
37 #include <android/hardware/audio/common/2.0/types.h>
39 #include "utility/AssertOk.h"
40 #include "utility/PrettyPrintAudioTypes.h"
41 #include "utility/ReturnIn.h"
48 using ::android::hardware::Return;
49 using ::android::hardware::hidl_handle;
50 using ::android::hardware::hidl_string;
51 using ::android::hardware::hidl_vec;
52 using ::android::hardware::MQDescriptorSync;
53 using ::android::hardware::audio::V2_0::AudioDrain;
54 using ::android::hardware::audio::V2_0::DeviceAddress;
55 using ::android::hardware::audio::V2_0::IDevice;
56 using ::android::hardware::audio::V2_0::IPrimaryDevice;
57 using TtyMode = ::android::hardware::audio::V2_0::IPrimaryDevice::TtyMode;
58 using ::android::hardware::audio::V2_0::IDevicesFactory;
59 using ::android::hardware::audio::V2_0::IStream;
60 using ::android::hardware::audio::V2_0::IStreamIn;
61 using ::android::hardware::audio::V2_0::TimeSpec;
62 using ReadParameters =
63 ::android::hardware::audio::V2_0::IStreamIn::ReadParameters;
64 using ReadStatus = ::android::hardware::audio::V2_0::IStreamIn::ReadStatus;
65 using ::android::hardware::audio::V2_0::IStreamOut;
66 using ::android::hardware::audio::V2_0::IStreamOutCallback;
67 using ::android::hardware::audio::V2_0::MmapBufferInfo;
68 using ::android::hardware::audio::V2_0::MmapPosition;
69 using ::android::hardware::audio::V2_0::ParameterValue;
70 using ::android::hardware::audio::V2_0::Result;
71 using ::android::hardware::audio::common::V2_0::AudioChannelMask;
72 using ::android::hardware::audio::common::V2_0::AudioConfig;
73 using ::android::hardware::audio::common::V2_0::AudioDevice;
74 using ::android::hardware::audio::common::V2_0::AudioFormat;
75 using ::android::hardware::audio::common::V2_0::AudioHandleConsts;
76 using ::android::hardware::audio::common::V2_0::AudioInputFlag;
77 using ::android::hardware::audio::common::V2_0::AudioIoHandle;
78 using ::android::hardware::audio::common::V2_0::AudioMode;
79 using ::android::hardware::audio::common::V2_0::AudioOffloadInfo;
80 using ::android::hardware::audio::common::V2_0::AudioOutputFlag;
81 using ::android::hardware::audio::common::V2_0::AudioSource;
82 using ::android::hardware::audio::common::V2_0::ThreadInfo;
84 using utility::returnIn;
86 const char* getTestName() {
87 return ::testing::UnitTest::GetInstance()->current_test_info()->name();
91 /** Document the current test case.
92 * Eg: calling `doc::test("Dump the state of the hal")` in the "debugDump" test
94 * <testcase name="debugDump" status="run" time="6"
95 * classname="AudioPrimaryHidlTest"
96 description="Dump the state of the hal." />
98 https://github.com/google/googletest/blob/master/googletest/docs/AdvancedGuide.md#logging-additional-information
100 void test(const std::string& testCaseDocumentation) {
101 ::testing::Test::RecordProperty("description", testCaseDocumentation);
104 /** Document why a test was not fully run. Usually due to an optional feature
105 * not implemented. */
106 void partialTest(const std::string& reason) {
107 LOG(INFO) << "Test " << getTestName() << " partially run: " << reason;
108 ::testing::Test::RecordProperty("partialyRunTest", reason);
111 /** Add a note to the test. */
112 void note(const std::string& note) {
113 LOG(INFO) << "Test " << getTestName() << " noted: " << note;
114 ::testing::Test::RecordProperty("note", note);
118 // Register callback for static object destruction
119 // Avoid destroying static objects after main return.
120 // Post main return destruction leads to incorrect gtest timing measurements as
122 // debuging if anything goes wrong during destruction.
123 class Environment : public ::testing::Environment {
125 using TearDownFunc = std::function<void()>;
126 void registerTearDown(TearDownFunc&& tearDown) {
127 tearDowns.push_back(std::move(tearDown));
131 void TearDown() override {
132 // Call the tear downs in reverse order of insertion
133 for (auto& tearDown : tearDowns) {
137 std::list<TearDownFunc> tearDowns;
139 // Instance to register global tearDown
140 static Environment* environment;
142 class HidlTest : public ::testing::VtsHalHidlTargetTestBase {
144 // Convenient member to store results
148 //////////////////////////////////////////////////////////////////////////////
149 ////////////////////// getService audio_devices_factory //////////////////////
150 //////////////////////////////////////////////////////////////////////////////
152 // Test all audio devices
153 class AudioHidlTest : public HidlTest {
155 void SetUp() override {
156 ASSERT_NO_FATAL_FAILURE(HidlTest::SetUp()); // setup base
158 if (devicesFactory == nullptr) {
159 environment->registerTearDown([] { devicesFactory.clear(); });
160 devicesFactory = ::testing::VtsHalHidlTargetTestBase::getService<
163 ASSERT_TRUE(devicesFactory != nullptr);
167 // Cache the devicesFactory retrieval to speed up each test by ~0.5s
168 static sp<IDevicesFactory> devicesFactory;
170 sp<IDevicesFactory> AudioHidlTest::devicesFactory;
172 TEST_F(AudioHidlTest, GetAudioDevicesFactoryService) {
173 doc::test("test the getService (called in SetUp)");
176 TEST_F(AudioHidlTest, OpenDeviceInvalidParameter) {
177 doc::test("test passing an invalid parameter to openDevice");
178 IDevicesFactory::Result result;
180 ASSERT_OK(devicesFactory->openDevice(IDevicesFactory::Device(-1),
181 returnIn(result, device)));
182 ASSERT_EQ(IDevicesFactory::Result::INVALID_ARGUMENTS, result);
183 ASSERT_TRUE(device == nullptr);
186 //////////////////////////////////////////////////////////////////////////////
187 /////////////////////////////// openDevice primary ///////////////////////////
188 //////////////////////////////////////////////////////////////////////////////
190 // Test the primary device
191 class AudioPrimaryHidlTest : public AudioHidlTest {
193 /** Primary HAL test are NOT thread safe. */
194 void SetUp() override {
195 ASSERT_NO_FATAL_FAILURE(AudioHidlTest::SetUp()); // setup base
197 if (device == nullptr) {
198 IDevicesFactory::Result result;
199 sp<IDevice> baseDevice;
201 devicesFactory->openDevice(IDevicesFactory::Device::PRIMARY,
202 returnIn(result, baseDevice)));
204 ASSERT_TRUE(baseDevice != nullptr);
206 environment->registerTearDown([] { device.clear(); });
207 device = IPrimaryDevice::castFrom(baseDevice);
208 ASSERT_TRUE(device != nullptr);
213 // Cache the device opening to speed up each test by ~0.5s
214 static sp<IPrimaryDevice> device;
216 sp<IPrimaryDevice> AudioPrimaryHidlTest::device;
218 TEST_F(AudioPrimaryHidlTest, OpenPrimaryDevice) {
219 doc::test("Test the openDevice (called in SetUp)");
222 TEST_F(AudioPrimaryHidlTest, Init) {
223 doc::test("Test that the audio primary hal initialized correctly");
224 ASSERT_OK(device->initCheck());
227 //////////////////////////////////////////////////////////////////////////////
228 ///////////////////// {set,get}{Master,Mic}{Mute,Volume} /////////////////////
229 //////////////////////////////////////////////////////////////////////////////
231 template <class Property>
232 class AccessorPrimaryHidlTest : public AudioPrimaryHidlTest {
234 /** Test a property getter and setter. */
235 template <class Getter, class Setter>
236 void testAccessors(const string& propertyName,
237 const vector<Property>& valuesToTest, Setter setter,
239 const vector<Property>& invalidValues = {}) {
240 Property initialValue; // Save initial value to restore it at the end
242 ASSERT_OK((device.get()->*getter)(returnIn(res, initialValue)));
245 for (Property setValue : valuesToTest) {
246 SCOPED_TRACE("Test " + propertyName + " getter and setter for " +
247 testing::PrintToString(setValue));
248 ASSERT_OK((device.get()->*setter)(setValue));
250 // Make sure the getter returns the same value just set
251 ASSERT_OK((device.get()->*getter)(returnIn(res, getValue)));
253 EXPECT_EQ(setValue, getValue);
256 for (Property invalidValue : invalidValues) {
257 SCOPED_TRACE("Try to set " + propertyName +
258 " with the invalid value " +
259 testing::PrintToString(invalidValue));
260 EXPECT_RESULT(Result::INVALID_ARGUMENTS,
261 (device.get()->*setter)(invalidValue));
265 (device.get()->*setter)(initialValue)); // restore initial value
268 /** Test the getter and setter of an optional feature. */
269 template <class Getter, class Setter>
270 void testOptionalAccessors(const string& propertyName,
271 const vector<Property>& valuesToTest,
272 Setter setter, Getter getter,
273 const vector<Property>& invalidValues = {}) {
274 doc::test("Test the optional " + propertyName + " getters and setter");
276 SCOPED_TRACE("Test feature support by calling the getter");
277 Property initialValue;
278 ASSERT_OK((device.get()->*getter)(returnIn(res, initialValue)));
279 if (res == Result::NOT_SUPPORTED) {
280 doc::partialTest(propertyName + " getter is not supported");
283 ASSERT_OK(res); // If it is supported it must succeed
285 // The feature is supported, test it
286 testAccessors(propertyName, valuesToTest, setter, getter,
291 using BoolAccessorPrimaryHidlTest = AccessorPrimaryHidlTest<bool>;
293 TEST_F(BoolAccessorPrimaryHidlTest, MicMuteTest) {
294 doc::test("Check that the mic can be muted and unmuted");
295 testAccessors("mic mute", {true, false, true}, &IDevice::setMicMute,
296 &IDevice::getMicMute);
297 // TODO: check that the mic is really muted (all sample are 0)
300 TEST_F(BoolAccessorPrimaryHidlTest, MasterMuteTest) {
302 "If master mute is supported, try to mute and unmute the master "
304 testOptionalAccessors("master mute", {true, false, true},
305 &IDevice::setMasterMute, &IDevice::getMasterMute);
306 // TODO: check that the master volume is really muted
309 using FloatAccessorPrimaryHidlTest = AccessorPrimaryHidlTest<float>;
310 TEST_F(FloatAccessorPrimaryHidlTest, MasterVolumeTest) {
311 doc::test("Test the master volume if supported");
312 testOptionalAccessors("master volume", {0, 0.5, 1},
313 &IDevice::setMasterVolume, &IDevice::getMasterVolume,
314 {-0.1, 1.1, NAN, INFINITY, -INFINITY,
315 1 + std::numeric_limits<float>::epsilon()});
316 // TODO: check that the master volume is really changed
319 //////////////////////////////////////////////////////////////////////////////
320 //////////////////////////////// AudioPatches ////////////////////////////////
321 //////////////////////////////////////////////////////////////////////////////
323 class AudioPatchPrimaryHidlTest : public AudioPrimaryHidlTest {
325 bool areAudioPatchesSupported() {
326 auto result = device->supportsAudioPatches();
327 EXPECT_IS_OK(result);
332 TEST_F(AudioPatchPrimaryHidlTest, AudioPatches) {
333 doc::test("Test if audio patches are supported");
334 if (!areAudioPatchesSupported()) {
335 doc::partialTest("Audio patches are not supported");
338 // TODO: test audio patches
341 //////////////////////////////////////////////////////////////////////////////
342 //////////////// Required and recommended audio format support ///////////////
344 // https://source.android.com/compatibility/android-cdd.html#5_4_audio_recording
346 // https://source.android.com/compatibility/android-cdd.html#5_5_audio_playback
347 /////////// TODO: move to the beginning of the file for easier update ////////
348 //////////////////////////////////////////////////////////////////////////////
350 class AudioConfigPrimaryTest : public AudioPatchPrimaryHidlTest {
353 static const vector<AudioConfig> getRequiredSupportPlaybackAudioConfig() {
354 return combineAudioConfig(
355 {AudioChannelMask::OUT_STEREO, AudioChannelMask::OUT_MONO},
356 {8000, 11025, 16000, 22050, 32000, 44100},
357 {AudioFormat::PCM_16_BIT});
360 static const vector<AudioConfig>
361 getRecommendedSupportPlaybackAudioConfig() {
362 return combineAudioConfig(
363 {AudioChannelMask::OUT_STEREO, AudioChannelMask::OUT_MONO},
364 {24000, 48000}, {AudioFormat::PCM_16_BIT});
367 static const vector<AudioConfig> getSupportedPlaybackAudioConfig() {
368 // TODO: retrieve audio config supported by the platform
369 // as declared in the policy configuration
373 static const vector<AudioConfig> getRequiredSupportCaptureAudioConfig() {
374 return combineAudioConfig({AudioChannelMask::IN_MONO},
375 {8000, 11025, 16000, 44100},
376 {AudioFormat::PCM_16_BIT});
378 static const vector<AudioConfig> getRecommendedSupportCaptureAudioConfig() {
379 return combineAudioConfig({AudioChannelMask::IN_STEREO}, {22050, 48000},
380 {AudioFormat::PCM_16_BIT});
382 static const vector<AudioConfig> getSupportedCaptureAudioConfig() {
383 // TODO: retrieve audio config supported by the platform
384 // as declared in the policy configuration
389 static const vector<AudioConfig> combineAudioConfig(
390 vector<AudioChannelMask> channelMasks, vector<uint32_t> sampleRates,
391 vector<AudioFormat> formats) {
392 vector<AudioConfig> configs;
393 for (auto channelMask : channelMasks) {
394 for (auto sampleRate : sampleRates) {
395 for (auto format : formats) {
396 AudioConfig config{};
397 // leave offloadInfo to 0
398 config.channelMask = channelMask;
399 config.sampleRateHz = sampleRate;
400 config.format = format;
401 // FIXME: leave frameCount to 0 ?
402 configs.push_back(config);
410 /** Generate a test name based on an audio config.
412 * As the only parameter changing are channel mask and sample rate,
413 * only print those ones in the test name.
415 static string generateTestName(
416 const testing::TestParamInfo<AudioConfig>& info) {
417 const AudioConfig& config = info.param;
418 return to_string(info.index) + "__" + to_string(config.sampleRateHz) + "_" +
419 // "MONO" is more clear than "FRONT_LEFT"
420 ((config.channelMask == AudioChannelMask::OUT_MONO ||
421 config.channelMask == AudioChannelMask::IN_MONO)
423 : toString(config.channelMask));
426 //////////////////////////////////////////////////////////////////////////////
427 ///////////////////////////// getInputBufferSize /////////////////////////////
428 //////////////////////////////////////////////////////////////////////////////
430 // FIXME: execute input test only if platform declares
431 // android.hardware.microphone
432 // how to get this value ? is it a property ???
434 class AudioCaptureConfigPrimaryTest
435 : public AudioConfigPrimaryTest,
436 public ::testing::WithParamInterface<AudioConfig> {
438 void inputBufferSizeTest(const AudioConfig& audioConfig,
439 bool supportRequired) {
442 device->getInputBufferSize(audioConfig, returnIn(res, bufferSize)));
445 case Result::INVALID_ARGUMENTS:
446 EXPECT_FALSE(supportRequired);
449 // Check that the buffer is of a sane size
450 // For now only that it is > 0
451 EXPECT_GT(bufferSize, uint64_t(0));
454 FAIL() << "Invalid return status: "
455 << ::testing::PrintToString(res);
460 // Test that the required capture config and those declared in the policy are
462 class RequiredInputBufferSizeTest : public AudioCaptureConfigPrimaryTest {};
463 TEST_P(RequiredInputBufferSizeTest, RequiredInputBufferSizeTest) {
465 "Input buffer size must be retrievable for a format with required "
467 inputBufferSizeTest(GetParam(), true);
469 INSTANTIATE_TEST_CASE_P(
470 RequiredInputBufferSize, RequiredInputBufferSizeTest,
472 AudioConfigPrimaryTest::getRequiredSupportCaptureAudioConfig()),
474 INSTANTIATE_TEST_CASE_P(
475 SupportedInputBufferSize, RequiredInputBufferSizeTest,
477 AudioConfigPrimaryTest::getSupportedCaptureAudioConfig()),
480 // Test that the recommended capture config are supported or lead to a
481 // INVALID_ARGUMENTS return
482 class OptionalInputBufferSizeTest : public AudioCaptureConfigPrimaryTest {};
483 TEST_P(OptionalInputBufferSizeTest, OptionalInputBufferSizeTest) {
485 "Input buffer size should be retrievable for a format with recommended "
487 inputBufferSizeTest(GetParam(), false);
489 INSTANTIATE_TEST_CASE_P(
490 RecommendedCaptureAudioConfigSupport, OptionalInputBufferSizeTest,
492 AudioConfigPrimaryTest::getRecommendedSupportCaptureAudioConfig()),
495 //////////////////////////////////////////////////////////////////////////////
496 /////////////////////////////// setScreenState ///////////////////////////////
497 //////////////////////////////////////////////////////////////////////////////
499 TEST_F(AudioPrimaryHidlTest, setScreenState) {
500 doc::test("Check that the hal can receive the screen state");
501 for (bool turnedOn : {false, true, true, false, false}) {
502 auto ret = device->setScreenState(turnedOn);
505 auto okOrNotSupported = {Result::OK, Result::NOT_SUPPORTED};
506 ASSERT_RESULT(okOrNotSupported, result);
510 //////////////////////////////////////////////////////////////////////////////
511 //////////////////////////// {get,set}Parameters /////////////////////////////
512 //////////////////////////////////////////////////////////////////////////////
514 TEST_F(AudioPrimaryHidlTest, getParameters) {
515 doc::test("Check that the hal can set and get parameters");
516 hidl_vec<hidl_string> keys;
517 hidl_vec<ParameterValue> values;
518 ASSERT_OK(device->getParameters(keys, returnIn(res, values)));
519 ASSERT_OK(device->setParameters(values));
521 ASSERT_OK(device->setParameters(values));
524 //////////////////////////////////////////////////////////////////////////////
525 //////////////////////////////// debugDebug //////////////////////////////////
526 //////////////////////////////////////////////////////////////////////////////
528 template <class DebugDump>
529 static void testDebugDump(DebugDump debugDump) {
530 // Dump in a temporary file
531 // Note that SELinux must be deactivate for this test to work
532 FILE* file = tmpfile();
533 ASSERT_NE(nullptr, file) << errno;
535 // Wrap the temporary file file descriptor in a native handle
536 auto* nativeHandle = native_handle_create(1, 0);
537 ASSERT_NE(nullptr, nativeHandle);
538 nativeHandle->data[0] = fileno(file);
540 // Wrap this native handle in a hidl handle
542 handle.setTo(nativeHandle, true /*take ownership*/);
544 ASSERT_OK(debugDump(handle));
546 // Check that at least one bit was written by the hal
547 // TODO: debugDump does not return a Result.
548 // This mean that the hal can not report that it not implementing the
550 rewind(file); // can not fail
552 if (fread(&buff, sizeof(buff), 1, file) != 1) {
553 doc::note("debugDump does not seem implemented");
555 EXPECT_EQ(0, fclose(file)) << errno;
558 TEST_F(AudioPrimaryHidlTest, DebugDump) {
559 doc::test("Check that the hal can dump its state without error");
561 [this](const auto& handle) { return device->debugDump(handle); });
564 TEST_F(AudioPrimaryHidlTest, DebugDumpInvalidArguments) {
565 doc::test("Check that the hal dump doesn't crash on invalid arguments");
566 ASSERT_OK(device->debugDump(hidl_handle()));
569 //////////////////////////////////////////////////////////////////////////////
570 ////////////////////////// open{Output,Input}Stream //////////////////////////
571 //////////////////////////////////////////////////////////////////////////////
573 template <class Stream>
574 class OpenStreamTest : public AudioConfigPrimaryTest,
575 public ::testing::WithParamInterface<AudioConfig> {
577 template <class Open>
578 void testOpen(Open openStream, const AudioConfig& config) {
579 // FIXME: Open a stream without an IOHandle
580 // This is not required to be accepted by hal implementations
581 AudioIoHandle ioHandle =
582 (AudioIoHandle)AudioHandleConsts::AUDIO_IO_HANDLE_NONE;
583 AudioConfig suggestedConfig{};
584 ASSERT_OK(openStream(ioHandle, config,
585 returnIn(res, stream, suggestedConfig)));
587 // TODO: only allow failure for RecommendedPlaybackAudioConfig
590 ASSERT_TRUE(stream != nullptr);
591 audioConfig = config;
593 case Result::INVALID_ARGUMENTS:
594 ASSERT_TRUE(stream == nullptr);
595 AudioConfig suggestedConfigRetry;
596 // Could not open stream with config, try again with the
599 openStream(ioHandle, suggestedConfig,
600 returnIn(res, stream, suggestedConfigRetry)));
601 // This time it must succeed
603 ASSERT_TRUE(stream != nullptr);
604 audioConfig = suggestedConfig;
607 FAIL() << "Invalid return status: "
608 << ::testing::PrintToString(res);
613 Return<Result> closeStream() {
615 return stream->close();
619 void TearDown() override {
621 ASSERT_OK(stream->close());
626 AudioConfig audioConfig;
627 DeviceAddress address = {};
632 ////////////////////////////// openOutputStream //////////////////////////////
634 class OutputStreamTest : public OpenStreamTest<IStreamOut> {
635 virtual void SetUp() override {
636 ASSERT_NO_FATAL_FAILURE(OpenStreamTest::SetUp()); // setup base
637 address.device = AudioDevice::OUT_DEFAULT;
638 const AudioConfig& config = GetParam();
639 AudioOutputFlag flags =
640 AudioOutputFlag::NONE; // TODO: test all flag combination
642 [&](AudioIoHandle handle, AudioConfig config, auto cb) {
643 return device->openOutputStream(handle, address, config, flags,
649 TEST_P(OutputStreamTest, OpenOutputStreamTest) {
651 "Check that output streams can be open with the required and "
652 "recommended config");
653 // Open done in SetUp
655 INSTANTIATE_TEST_CASE_P(
656 RequiredOutputStreamConfigSupport, OutputStreamTest,
658 AudioConfigPrimaryTest::getRequiredSupportPlaybackAudioConfig()),
660 INSTANTIATE_TEST_CASE_P(
661 SupportedOutputStreamConfig, OutputStreamTest,
663 AudioConfigPrimaryTest::getSupportedPlaybackAudioConfig()),
666 INSTANTIATE_TEST_CASE_P(
667 RecommendedOutputStreamConfigSupport, OutputStreamTest,
669 AudioConfigPrimaryTest::getRecommendedSupportPlaybackAudioConfig()),
672 ////////////////////////////// openInputStream //////////////////////////////
674 class InputStreamTest : public OpenStreamTest<IStreamIn> {
675 virtual void SetUp() override {
676 ASSERT_NO_FATAL_FAILURE(OpenStreamTest::SetUp()); // setup base
677 address.device = AudioDevice::IN_DEFAULT;
678 const AudioConfig& config = GetParam();
679 AudioInputFlag flags =
680 AudioInputFlag::NONE; // TODO: test all flag combination
682 AudioSource::DEFAULT; // TODO: test all flag combination
684 [&](AudioIoHandle handle, AudioConfig config, auto cb) {
685 return device->openInputStream(handle, address, config, flags,
692 TEST_P(InputStreamTest, OpenInputStreamTest) {
694 "Check that input streams can be open with the required and "
695 "recommended config");
696 // Open done in setup
698 INSTANTIATE_TEST_CASE_P(
699 RequiredInputStreamConfigSupport, InputStreamTest,
701 AudioConfigPrimaryTest::getRequiredSupportCaptureAudioConfig()),
703 INSTANTIATE_TEST_CASE_P(
704 SupportedInputStreamConfig, InputStreamTest,
706 AudioConfigPrimaryTest::getSupportedCaptureAudioConfig()),
709 INSTANTIATE_TEST_CASE_P(
710 RecommendedInputStreamConfigSupport, InputStreamTest,
712 AudioConfigPrimaryTest::getRecommendedSupportCaptureAudioConfig()),
715 //////////////////////////////////////////////////////////////////////////////
716 ////////////////////////////// IStream getters ///////////////////////////////
717 //////////////////////////////////////////////////////////////////////////////
719 /** Unpack the provided result.
720 * If the result is not OK, register a failure and return an undefined value. */
722 static R extract(Return<R> ret) {
730 /* Could not find a way to write a test for two parametrized class fixure
731 * thus use this macro do duplicate tests for Input and Output stream */
732 #define TEST_IO_STREAM(test_name, documentation, code) \
733 TEST_P(InputStreamTest, test_name) { \
734 doc::test(documentation); \
737 TEST_P(OutputStreamTest, test_name) { \
738 doc::test(documentation); \
744 "Check that the stream frame count == the one it was opened with",
745 ASSERT_EQ(audioConfig.frameCount, extract(stream->getFrameCount())))
749 "Check that the stream sample rate == the one it was opened with",
750 stream->getSampleRate())
754 "Check that the stream channel mask == the one it was opened with",
755 stream->getChannelMask())
757 TEST_IO_STREAM(GetFormat,
758 "Check that the stream format == the one it was opened with",
759 ASSERT_EQ(audioConfig.format, extract(stream->getFormat())))
761 // TODO: for now only check that the framesize is not incoherent
762 TEST_IO_STREAM(GetFrameSize,
763 "Check that the stream frame size == the one it was opened with",
764 ASSERT_GT(extract(stream->getFrameSize()), 0U))
766 TEST_IO_STREAM(GetBufferSize,
767 "Check that the stream buffer size== the one it was opened with",
768 ASSERT_GE(extract(stream->getBufferSize()),
769 extract(stream->getFrameSize())));
771 template <class Property, class CapabilityGetter, class Getter, class Setter>
772 static void testCapabilityGetter(const string& name, IStream* stream,
773 Property currentValue,
774 CapabilityGetter capablityGetter,
775 Getter getter, Setter setter) {
776 hidl_vec<Property> capabilities;
777 ASSERT_OK((stream->*capablityGetter)(returnIn(capabilities)));
778 if (capabilities.size() == 0) {
779 // The default hal should probably return a NOT_SUPPORTED if the hal
781 // capability retrieval. For now it returns an empty list if not
783 doc::partialTest(name + " is not supported");
786 // TODO: This code has never been tested on a hal that supports
787 // getSupportedSampleRates
788 EXPECT_NE(std::find(capabilities.begin(), capabilities.end(), currentValue),
790 << "current " << name << " is not in the list of the supported ones "
791 << toString(capabilities);
793 // Check that all declared supported values are indeed supported
794 for (auto capability : capabilities) {
795 ASSERT_OK((stream->*setter)(capability));
796 ASSERT_EQ(capability, extract((stream->*getter)()));
800 TEST_IO_STREAM(SupportedSampleRate,
801 "Check that the stream sample rate is declared as supported",
802 testCapabilityGetter("getSupportedSampleRate", stream.get(),
803 extract(stream->getSampleRate()),
804 &IStream::getSupportedSampleRates,
805 &IStream::getSampleRate,
806 &IStream::setSampleRate))
808 TEST_IO_STREAM(SupportedChannelMask,
809 "Check that the stream channel mask is declared as supported",
810 testCapabilityGetter("getSupportedChannelMask", stream.get(),
811 extract(stream->getChannelMask()),
812 &IStream::getSupportedChannelMasks,
813 &IStream::getChannelMask,
814 &IStream::setChannelMask))
816 TEST_IO_STREAM(SupportedFormat,
817 "Check that the stream format is declared as supported",
818 testCapabilityGetter("getSupportedFormat", stream.get(),
819 extract(stream->getFormat()),
820 &IStream::getSupportedFormats,
821 &IStream::getFormat, &IStream::setFormat))
823 static void testGetDevice(IStream* stream, AudioDevice expectedDevice) {
824 // Unfortunately the interface does not allow the implementation to return
826 // Thus allow NONE as signaling that the call is not supported.
827 auto ret = stream->getDevice();
829 AudioDevice device = ret;
830 ASSERT_TRUE(device == expectedDevice || device == AudioDevice::NONE)
831 << "Expected: " << ::testing::PrintToString(expectedDevice)
832 << "\n Actual: " << ::testing::PrintToString(device);
835 TEST_IO_STREAM(GetDevice,
836 "Check that the stream device == the one it was opened with",
837 areAudioPatchesSupported()
838 ? doc::partialTest("Audio patches are supported")
839 : testGetDevice(stream.get(), address.device))
841 static void testSetDevice(IStream* stream, const DeviceAddress& address) {
842 DeviceAddress otherAddress = address;
843 otherAddress.device = (address.device & AudioDevice::BIT_IN) == 0
844 ? AudioDevice::OUT_SPEAKER
845 : AudioDevice::IN_BUILTIN_MIC;
846 EXPECT_OK(stream->setDevice(otherAddress));
848 ASSERT_OK(stream->setDevice(address)); // Go back to the original value
853 "Check that the stream can be rerouted to SPEAKER or BUILTIN_MIC",
854 areAudioPatchesSupported() ? doc::partialTest("Audio patches are supported")
855 : testSetDevice(stream.get(), address))
857 static void testGetAudioProperties(IStream* stream) {
858 uint32_t sampleRateHz;
859 AudioChannelMask mask;
861 stream->getAudioProperties(returnIn(sampleRateHz, mask, format));
866 "Check that the stream audio properties == the ones it was opened with",
867 testGetAudioProperties(stream.get()))
869 static void testConnectedState(IStream* stream) {
870 DeviceAddress address = {};
871 using AD = AudioDevice;
873 {AD::OUT_HDMI, AD::OUT_WIRED_HEADPHONE, AD::IN_USB_HEADSET}) {
874 address.device = device;
876 ASSERT_OK(stream->setConnectedState(address, true));
877 ASSERT_OK(stream->setConnectedState(address, false));
880 TEST_IO_STREAM(SetConnectedState,
881 "Check that the stream can be notified of device connection and "
883 testConnectedState(stream.get()))
885 static auto invalidArgsOrNotSupportedOrOK = {Result::INVALID_ARGUMENTS,
886 Result::NOT_SUPPORTED, Result::OK};
887 TEST_IO_STREAM(SetHwAvSync, "Try to set hardware sync to an invalid value",
888 ASSERT_RESULT(invalidArgsOrNotSupportedOrOK,
889 stream->setHwAvSync(666)))
891 TEST_IO_STREAM(GetHwAvSync, "Get hardware sync can not fail",
892 ASSERT_IS_OK(device->getHwAvSync()));
894 static void checkGetNoParameter(IStream* stream, hidl_vec<hidl_string> keys,
895 vector<Result> expectedResults) {
896 hidl_vec<ParameterValue> parameters;
898 ASSERT_OK(stream->getParameters(keys, returnIn(res, parameters)));
899 ASSERT_RESULT(expectedResults, res);
900 if (res == Result::OK) {
901 for (auto& parameter : parameters) {
902 ASSERT_EQ(0U, parameter.value.size()) << toString(parameter);
907 /* Get/Set parameter is intended to be an opaque channel between vendors app and
909 * Thus can not be meaningfully tested.
911 TEST_IO_STREAM(getEmptySetParameter, "Retrieve the values of an empty set",
912 checkGetNoParameter(stream.get(), {} /* keys */, {Result::OK}))
914 TEST_IO_STREAM(getNonExistingParameter,
915 "Retrieve the values of an non existing parameter",
916 checkGetNoParameter(stream.get(),
917 {"Non existing key"} /* keys */,
918 {Result::NOT_SUPPORTED}))
920 TEST_IO_STREAM(setEmptySetParameter,
921 "Set the values of an empty set of parameters",
922 ASSERT_RESULT(Result::OK, stream->setParameters({})))
925 setNonExistingParameter, "Set the values of an non existing parameter",
926 // Unfortunately, the set_parameter legacy interface did not return any
927 // error code when a key is not supported.
928 // To allow implementation to just wrapped the legacy one, consider OK as a
929 // valid result for setting a non existing parameter.
930 ASSERT_RESULT(invalidArgsOrNotSupportedOrOK,
931 stream->setParameters({{"non existing key", "0"}})))
933 TEST_IO_STREAM(DebugDump,
934 "Check that a stream can dump its state without error",
935 testDebugDump([this](const auto& handle) {
936 return stream->debugDump(handle);
939 TEST_IO_STREAM(DebugDumpInvalidArguments,
940 "Check that the stream dump doesn't crash on invalid arguments",
941 ASSERT_OK(stream->debugDump(hidl_handle())))
943 //////////////////////////////////////////////////////////////////////////////
944 ////////////////////////////// addRemoveEffect ///////////////////////////////
945 //////////////////////////////////////////////////////////////////////////////
947 TEST_IO_STREAM(AddNonExistingEffect, "Adding a non existing effect should fail",
948 ASSERT_RESULT(Result::INVALID_ARGUMENTS, stream->addEffect(666)))
949 TEST_IO_STREAM(RemoveNonExistingEffect,
950 "Removing a non existing effect should fail",
951 ASSERT_RESULT(Result::INVALID_ARGUMENTS,
952 stream->removeEffect(666)))
954 // TODO: positive tests
956 //////////////////////////////////////////////////////////////////////////////
957 /////////////////////////////// Control ////////////////////////////////
958 //////////////////////////////////////////////////////////////////////////////
960 TEST_IO_STREAM(standby, "Make sure the stream can be put in stanby",
961 ASSERT_OK(stream->standby())) // can not fail
963 static vector<Result> invalidStateOrNotSupported = {Result::INVALID_STATE,
964 Result::NOT_SUPPORTED};
966 TEST_IO_STREAM(startNoMmap,
967 "Starting a mmaped stream before mapping it should fail",
968 ASSERT_RESULT(invalidStateOrNotSupported, stream->start()))
970 TEST_IO_STREAM(stopNoMmap,
971 "Stopping a mmaped stream before mapping it should fail",
972 ASSERT_RESULT(invalidStateOrNotSupported, stream->stop()))
974 TEST_IO_STREAM(getMmapPositionNoMmap,
975 "Get a stream Mmap position before mapping it should fail",
976 ASSERT_RESULT(invalidStateOrNotSupported, stream->stop()))
978 TEST_IO_STREAM(close, "Make sure a stream can be closed",
979 ASSERT_OK(closeStream()))
980 TEST_IO_STREAM(closeTwice, "Make sure a stream can not be closed twice",
981 ASSERT_OK(closeStream());
982 ASSERT_RESULT(Result::INVALID_STATE, closeStream()))
984 static auto invalidArgsOrNotSupported = {Result::INVALID_ARGUMENTS,
985 Result::NOT_SUPPORTED};
986 static void testCreateTooBigMmapBuffer(IStream* stream) {
989 // Assume that int max is a value too big to be allocated
990 // This is true currently with a 32bit media server, but might not when it
991 // will run in 64 bit
992 auto minSizeFrames = std::numeric_limits<int32_t>::max();
993 ASSERT_OK(stream->createMmapBuffer(minSizeFrames, returnIn(res, info)));
994 ASSERT_RESULT(invalidArgsOrNotSupported, res);
997 TEST_IO_STREAM(CreateTooBigMmapBuffer, "Create mmap buffer too big should fail",
998 testCreateTooBigMmapBuffer(stream.get()))
1000 static void testGetMmapPositionOfNonMmapedStream(IStream* stream) {
1002 MmapPosition position;
1003 ASSERT_OK(stream->getMmapPosition(returnIn(res, position)));
1004 ASSERT_RESULT(invalidArgsOrNotSupported, res);
1008 GetMmapPositionOfNonMmapedStream,
1009 "Retrieving the mmap position of a non mmaped stream should fail",
1010 testGetMmapPositionOfNonMmapedStream(stream.get()))
1012 //////////////////////////////////////////////////////////////////////////////
1013 ///////////////////////////////// StreamIn ///////////////////////////////////
1014 //////////////////////////////////////////////////////////////////////////////
1016 TEST_P(InputStreamTest, GetAudioSource) {
1018 "Retrieving the audio source of an input stream should always succeed");
1020 ASSERT_OK(stream->getAudioSource(returnIn(res, source)));
1021 if (res == Result::NOT_SUPPORTED) {
1022 doc::partialTest("getAudioSource is not supported");
1026 ASSERT_EQ(AudioSource::DEFAULT, source);
1029 static void testUnitaryGain(std::function<Return<Result>(float)> setGain) {
1031 (float[]){-INFINITY, -1.0, 1.0 + std::numeric_limits<float>::epsilon(),
1032 2.0, INFINITY, NAN}) {
1033 EXPECT_RESULT(Result::INVALID_ARGUMENTS, setGain(value)) << "value="
1036 // Do not consider -0.0 as an invalid value as it is == with 0.0
1037 for (float value : {-0.0, 0.0, 0.01, 0.5, 0.09, 1.0 /* Restore volume*/}) {
1038 EXPECT_OK(setGain(value)) << "value=" << value;
1042 static void testOptionalUnitaryGain(
1043 std::function<Return<Result>(float)> setGain, string debugName) {
1044 auto result = setGain(1);
1045 ASSERT_IS_OK(result);
1046 if (result == Result::NOT_SUPPORTED) {
1047 doc::partialTest(debugName + " is not supported");
1050 testUnitaryGain(setGain);
1053 TEST_P(InputStreamTest, SetGain) {
1054 doc::test("The gain of an input stream should only be set between [0,1]");
1055 testOptionalUnitaryGain(
1056 [this](float volume) { return stream->setGain(volume); },
1057 "InputStream::setGain");
1060 static void testPrepareForReading(IStreamIn* stream, uint32_t frameSize, uint32_t framesCount,
1061 bool allowSucceed) {
1063 // Ignore output parameters.
1064 ASSERT_OK(stream->prepareForReading(
1065 frameSize, framesCount,
1066 [&res](auto r, auto&, auto&, auto&, auto&) { res = r; }));
1069 Result::INVALID_ARGUMENTS, Result::OK,
1071 EXPECT_RESULT(status, res);
1073 EXPECT_RESULT(Result::INVALID_ARGUMENTS, res);
1077 TEST_P(InputStreamTest, PrepareForReadingWithZeroBuffer) {
1079 "Preparing a stream for reading with a 0 sized buffer should fail");
1080 testPrepareForReading(stream.get(), 0, 0, false /*allowSucceed*/);
1083 TEST_P(InputStreamTest, PrepareForReadingWithHugeBuffer) {
1085 "Preparing a stream for reading with a 2^32 sized buffer should fail");
1086 testPrepareForReading(stream.get(), 1, std::numeric_limits<uint32_t>::max(),
1087 false /*allowSucceed*/);
1090 TEST_P(InputStreamTest, PrepareForReadingCheckOverflow) {
1092 "Preparing a stream for reading with a overflowing sized buffer should "
1094 auto uintMax = std::numeric_limits<uint32_t>::max();
1095 // In O, the test fails for 32-bit HAL, and succeeds for 64-bit HAL.
1096 testPrepareForReading(stream.get(), uintMax, uintMax, true /*allowSucceed*/);
1099 TEST_P(InputStreamTest, GetInputFramesLost) {
1101 "The number of frames lost on a never started stream should be 0");
1102 auto ret = stream->getInputFramesLost();
1104 uint32_t framesLost{ret};
1105 ASSERT_EQ(0U, framesLost);
1108 TEST_P(InputStreamTest, getCapturePosition) {
1110 "The capture position of a non prepared stream should not be "
1114 ASSERT_OK(stream->getCapturePosition(returnIn(res, frames, time)));
1115 ASSERT_RESULT(invalidStateOrNotSupported, res);
1118 //////////////////////////////////////////////////////////////////////////////
1119 ///////////////////////////////// StreamIn ///////////////////////////////////
1120 //////////////////////////////////////////////////////////////////////////////
1122 TEST_P(OutputStreamTest, getLatency) {
1123 doc::test("Make sure latency is over 0");
1124 auto result = stream->getLatency();
1125 ASSERT_IS_OK(result);
1126 ASSERT_GT(result, 0U);
1129 TEST_P(OutputStreamTest, setVolume) {
1130 doc::test("Try to set the output volume");
1131 testOptionalUnitaryGain(
1132 [this](float volume) { return stream->setVolume(volume, volume); },
1136 static void testPrepareForWriting(IStreamOut* stream, uint32_t frameSize, uint32_t framesCount,
1137 bool allowSucceed) {
1139 // Ignore output parameters.
1140 ASSERT_OK(stream->prepareForWriting(
1141 frameSize, framesCount,
1142 [&res](auto r, auto&, auto&, auto&, auto&) { res = r; }));
1145 Result::INVALID_ARGUMENTS, Result::OK,
1147 EXPECT_RESULT(status, res);
1149 EXPECT_RESULT(Result::INVALID_ARGUMENTS, res);
1153 TEST_P(OutputStreamTest, PrepareForWriteWithZeroBuffer) {
1155 "Preparing a stream for writing with a 0 sized buffer should fail");
1156 testPrepareForWriting(stream.get(), 0, 0, false /*allowSucceed*/);
1159 TEST_P(OutputStreamTest, PrepareForWriteWithHugeBuffer) {
1161 "Preparing a stream for writing with a 2^32 sized buffer should fail");
1162 testPrepareForWriting(stream.get(), 1, std::numeric_limits<uint32_t>::max(),
1163 false /*allowSucceed*/);
1166 TEST_P(OutputStreamTest, PrepareForWritingCheckOverflow) {
1168 "Preparing a stream for writing with a overflowing sized buffer should "
1170 auto uintMax = std::numeric_limits<uint32_t>::max();
1171 // In O, the test fails for 32-bit HAL, and succeeds for 64-bit HAL.
1172 testPrepareForWriting(stream.get(), uintMax, uintMax, true /*allowSucceed*/);
1176 Capability(IStreamOut* stream) {
1177 EXPECT_OK(stream->supportsPauseAndResume(returnIn(pause, resume)));
1178 auto ret = stream->supportsDrain();
1185 bool resume = false;
1189 TEST_P(OutputStreamTest, SupportsPauseAndResumeAndDrain) {
1191 "Implementation must expose pause, resume and drain capabilities");
1192 Capability(stream.get());
1195 template <class Value>
1196 static void checkInvalidStateOr0(Result res, Value value) {
1198 case Result::INVALID_STATE:
1201 ASSERT_EQ(0U, value);
1204 FAIL() << "Unexpected result " << toString(res);
1208 TEST_P(OutputStreamTest, GetRenderPosition) {
1209 doc::test("A new stream render position should be 0 or INVALID_STATE");
1211 ASSERT_OK(stream->getRenderPosition(returnIn(res, dspFrames)));
1212 if (res == Result::NOT_SUPPORTED) {
1213 doc::partialTest("getRenderPosition is not supported");
1216 checkInvalidStateOr0(res, dspFrames);
1219 TEST_P(OutputStreamTest, GetNextWriteTimestamp) {
1220 doc::test("A new stream next write timestamp should be 0 or INVALID_STATE");
1221 uint64_t timestampUs;
1222 ASSERT_OK(stream->getNextWriteTimestamp(returnIn(res, timestampUs)));
1223 if (res == Result::NOT_SUPPORTED) {
1224 doc::partialTest("getNextWriteTimestamp is not supported");
1227 checkInvalidStateOr0(res, timestampUs);
1230 /** Stub implementation of out stream callback. */
1231 class MockOutCallbacks : public IStreamOutCallback {
1232 Return<void> onWriteReady() override { return {}; }
1233 Return<void> onDrainReady() override { return {}; }
1234 Return<void> onError() override { return {}; }
1237 static bool isAsyncModeSupported(IStreamOut* stream) {
1238 auto res = stream->setCallback(new MockOutCallbacks);
1239 stream->clearCallback(); // try to restore the no callback state, ignore
1241 auto okOrNotSupported = {Result::OK, Result::NOT_SUPPORTED};
1242 EXPECT_RESULT(okOrNotSupported, res);
1243 return res.isOk() ? res == Result::OK : false;
1246 TEST_P(OutputStreamTest, SetCallback) {
1248 "If supported, registering callback for async operation should never "
1250 if (!isAsyncModeSupported(stream.get())) {
1251 doc::partialTest("The stream does not support async operations");
1254 ASSERT_OK(stream->setCallback(new MockOutCallbacks));
1255 ASSERT_OK(stream->setCallback(new MockOutCallbacks));
1258 TEST_P(OutputStreamTest, clearCallback) {
1260 "If supported, clearing a callback to go back to sync operation should "
1262 if (!isAsyncModeSupported(stream.get())) {
1263 doc::partialTest("The stream does not support async operations");
1266 // TODO: Clarify if clearing a non existing callback should fail
1267 ASSERT_OK(stream->setCallback(new MockOutCallbacks));
1268 ASSERT_OK(stream->clearCallback());
1271 TEST_P(OutputStreamTest, Resume) {
1273 "If supported, a stream should fail to resume if not previously "
1275 if (!Capability(stream.get()).resume) {
1276 doc::partialTest("The output stream does not support resume");
1279 ASSERT_RESULT(Result::INVALID_STATE, stream->resume());
1282 TEST_P(OutputStreamTest, Pause) {
1284 "If supported, a stream should fail to pause if not previously "
1286 if (!Capability(stream.get()).pause) {
1287 doc::partialTest("The output stream does not support pause");
1290 ASSERT_RESULT(Result::INVALID_STATE, stream->resume());
1293 static void testDrain(IStreamOut* stream, AudioDrain type) {
1294 if (!Capability(stream).drain) {
1295 doc::partialTest("The output stream does not support drain");
1298 ASSERT_RESULT(Result::OK, stream->drain(type));
1301 TEST_P(OutputStreamTest, DrainAll) {
1302 doc::test("If supported, a stream should always succeed to drain");
1303 testDrain(stream.get(), AudioDrain::ALL);
1306 TEST_P(OutputStreamTest, DrainEarlyNotify) {
1307 doc::test("If supported, a stream should always succeed to drain");
1308 testDrain(stream.get(), AudioDrain::EARLY_NOTIFY);
1311 TEST_P(OutputStreamTest, FlushStop) {
1312 doc::test("If supported, a stream should always succeed to flush");
1313 auto ret = stream->flush();
1315 if (ret == Result::NOT_SUPPORTED) {
1316 doc::partialTest("Flush is not supported");
1322 TEST_P(OutputStreamTest, GetPresentationPositionStop) {
1324 "If supported, a stream should always succeed to retrieve the "
1325 "presentation position");
1328 ASSERT_OK(stream->getPresentationPosition(returnIn(res, frames, mesureTS)));
1329 if (res == Result::NOT_SUPPORTED) {
1330 doc::partialTest("getpresentationPosition is not supported");
1333 ASSERT_EQ(0U, frames);
1335 if (mesureTS.tvNSec == 0 && mesureTS.tvSec == 0) {
1336 // As the stream has never written a frame yet,
1337 // the timestamp does not really have a meaning, allow to return 0
1341 // Make sure the return measure is not more than 1s old.
1342 struct timespec currentTS;
1343 ASSERT_EQ(0, clock_gettime(CLOCK_MONOTONIC, ¤tTS)) << errno;
1345 auto toMicroSec = [](uint64_t sec, auto nsec) {
1346 return sec * 1e+6 + nsec / 1e+3;
1348 auto currentTime = toMicroSec(currentTS.tv_sec, currentTS.tv_nsec);
1349 auto mesureTime = toMicroSec(mesureTS.tvSec, mesureTS.tvNSec);
1350 ASSERT_PRED2([](auto c, auto m) { return c - m < 1e+6; }, currentTime,
1354 //////////////////////////////////////////////////////////////////////////////
1355 /////////////////////////////// PrimaryDevice ////////////////////////////////
1356 //////////////////////////////////////////////////////////////////////////////
1358 TEST_F(AudioPrimaryHidlTest, setVoiceVolume) {
1359 doc::test("Make sure setVoiceVolume only succeed if volume is in [0,1]");
1361 [this](float volume) { return device->setVoiceVolume(volume); });
1364 TEST_F(AudioPrimaryHidlTest, setMode) {
1366 "Make sure setMode always succeeds if mode is valid "
1367 "and fails otherwise");
1368 // Test Invalid values
1369 for (AudioMode mode :
1370 {AudioMode::INVALID, AudioMode::CURRENT, AudioMode::CNT}) {
1371 SCOPED_TRACE("mode=" + toString(mode));
1372 ASSERT_RESULT(Result::INVALID_ARGUMENTS, device->setMode(mode));
1374 // Test valid values
1375 for (AudioMode mode :
1376 {AudioMode::IN_CALL, AudioMode::IN_COMMUNICATION, AudioMode::RINGTONE,
1377 AudioMode::NORMAL /* Make sure to leave the test in normal mode */}) {
1378 SCOPED_TRACE("mode=" + toString(mode));
1379 ASSERT_OK(device->setMode(mode));
1383 TEST_F(BoolAccessorPrimaryHidlTest, BtScoNrecEnabled) {
1384 doc::test("Query and set the BT SCO NR&EC state");
1385 testOptionalAccessors("BtScoNrecEnabled", {true, false, true},
1386 &IPrimaryDevice::setBtScoNrecEnabled,
1387 &IPrimaryDevice::getBtScoNrecEnabled);
1390 TEST_F(BoolAccessorPrimaryHidlTest, setGetBtScoWidebandEnabled) {
1391 doc::test("Query and set the SCO whideband state");
1392 testOptionalAccessors("BtScoWideband", {true, false, true},
1393 &IPrimaryDevice::setBtScoWidebandEnabled,
1394 &IPrimaryDevice::getBtScoWidebandEnabled);
1397 using TtyModeAccessorPrimaryHidlTest = AccessorPrimaryHidlTest<TtyMode>;
1398 TEST_F(TtyModeAccessorPrimaryHidlTest, setGetTtyMode) {
1399 doc::test("Query and set the TTY mode state");
1400 testOptionalAccessors(
1401 "TTY mode", {TtyMode::OFF, TtyMode::HCO, TtyMode::VCO, TtyMode::FULL},
1402 &IPrimaryDevice::setTtyMode, &IPrimaryDevice::getTtyMode);
1405 TEST_F(BoolAccessorPrimaryHidlTest, setGetHac) {
1406 doc::test("Query and set the HAC state");
1407 testOptionalAccessors("HAC", {true, false, true},
1408 &IPrimaryDevice::setHacEnabled,
1409 &IPrimaryDevice::getHacEnabled);
1412 //////////////////////////////////////////////////////////////////////////////
1413 //////////////////// Clean caches on global tear down ////////////////////////
1414 //////////////////////////////////////////////////////////////////////////////
1416 int main(int argc, char** argv) {
1417 environment = new Environment;
1418 ::testing::AddGlobalTestEnvironment(environment);
1419 ::testing::InitGoogleTest(&argc, argv);
1420 int status = RUN_ALL_TESTS();
1421 LOG(INFO) << "Test result = " << status;