• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2016 The Android Open Source Project
3  *
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
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
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.
15  */
16 
17 #define LOG_TAG "SoundTriggerHidlHalTest"
18 #include <stdlib.h>
19 #include <time.h>
20 
21 #include <condition_variable>
22 #include <mutex>
23 
24 #include <android/log.h>
25 #include <cutils/native_handle.h>
26 #include <log/log.h>
27 
28 #include <android/hardware/audio/common/2.0/types.h>
29 #include <android/hardware/soundtrigger/2.0/ISoundTriggerHw.h>
30 #include <android/hardware/soundtrigger/2.0/types.h>
31 
32 #include <VtsHalHidlTargetTestBase.h>
33 #include <VtsHalHidlTargetTestEnvBase.h>
34 
35 #define SHORT_TIMEOUT_PERIOD (1)
36 
37 using ::android::hardware::audio::common::V2_0::AudioDevice;
38 using ::android::hardware::soundtrigger::V2_0::SoundModelHandle;
39 using ::android::hardware::soundtrigger::V2_0::SoundModelType;
40 using ::android::hardware::soundtrigger::V2_0::RecognitionMode;
41 using ::android::hardware::soundtrigger::V2_0::PhraseRecognitionExtra;
42 using ::android::hardware::soundtrigger::V2_0::ISoundTriggerHw;
43 using ::android::hardware::soundtrigger::V2_0::ISoundTriggerHwCallback;
44 using ::android::hardware::Return;
45 using ::android::hardware::Void;
46 using ::android::sp;
47 
48 /**
49  * Test code uses this class to wait for notification from callback.
50  */
51 class Monitor {
52  public:
Monitor()53   Monitor() : mCount(0) {}
54 
55   /**
56    * Adds 1 to the internal counter and unblocks one of the waiting threads.
57    */
notify()58   void notify() {
59     std::unique_lock<std::mutex> lock(mMtx);
60     mCount++;
61     mCv.notify_one();
62   }
63 
64   /**
65    * Blocks until the internal counter becomes greater than 0.
66    *
67    * If notified, this method decreases the counter by 1 and returns true.
68    * If timeout, returns false.
69    */
wait(int timeoutSeconds)70   bool wait(int timeoutSeconds) {
71     std::unique_lock<std::mutex> lock(mMtx);
72     auto deadline = std::chrono::system_clock::now() +
73         std::chrono::seconds(timeoutSeconds);
74     while (mCount == 0) {
75       if (mCv.wait_until(lock, deadline) == std::cv_status::timeout) {
76         return false;
77       }
78     }
79     mCount--;
80     return true;
81   }
82 
83  private:
84   std::mutex mMtx;
85   std::condition_variable mCv;
86   int mCount;
87 };
88 
89 // Test environment for SoundTrigger HIDL HAL.
90 class SoundTriggerHidlEnvironment : public ::testing::VtsHalHidlTargetTestEnvBase {
91    public:
92     // get the test environment singleton
Instance()93     static SoundTriggerHidlEnvironment* Instance() {
94         static SoundTriggerHidlEnvironment* instance = new SoundTriggerHidlEnvironment;
95         return instance;
96     }
97 
registerTestServices()98     virtual void registerTestServices() override { registerTestService<ISoundTriggerHw>(); }
99 
100    private:
SoundTriggerHidlEnvironment()101     SoundTriggerHidlEnvironment() {}
102 };
103 
104 // The main test class for Sound Trigger HIDL HAL.
105 class SoundTriggerHidlTest : public ::testing::VtsHalHidlTargetTestBase {
106  public:
SetUp()107   virtual void SetUp() override {
108       mSoundTriggerHal = ::testing::VtsHalHidlTargetTestBase::getService<ISoundTriggerHw>(
109           SoundTriggerHidlEnvironment::Instance()->getServiceName<ISoundTriggerHw>());
110       ASSERT_NE(nullptr, mSoundTriggerHal.get());
111       mCallback = new SoundTriggerHwCallback(*this);
112       ASSERT_NE(nullptr, mCallback.get());
113   }
114 
SetUpTestCase()115   static void SetUpTestCase() {
116     srand(time(nullptr));
117   }
118 
119   class SoundTriggerHwCallback : public ISoundTriggerHwCallback {
120    private:
121     SoundTriggerHidlTest& mParent;
122 
123    public:
SoundTriggerHwCallback(SoundTriggerHidlTest & parent)124     SoundTriggerHwCallback(SoundTriggerHidlTest& parent) : mParent(parent) {}
125 
recognitionCallback(const ISoundTriggerHwCallback::RecognitionEvent & event __unused,int32_t cookie __unused)126     virtual Return<void> recognitionCallback(
127         const ISoundTriggerHwCallback::RecognitionEvent& event __unused,
128         int32_t cookie __unused) {
129       ALOGI("%s", __FUNCTION__);
130       return Void();
131     }
132 
phraseRecognitionCallback(const ISoundTriggerHwCallback::PhraseRecognitionEvent & event __unused,int32_t cookie __unused)133     virtual Return<void> phraseRecognitionCallback(
134         const ISoundTriggerHwCallback::PhraseRecognitionEvent& event __unused,
135         int32_t cookie __unused) {
136       ALOGI("%s", __FUNCTION__);
137       return Void();
138     }
139 
soundModelCallback(const ISoundTriggerHwCallback::ModelEvent & event,int32_t cookie __unused)140     virtual Return<void> soundModelCallback(
141         const ISoundTriggerHwCallback::ModelEvent& event,
142         int32_t cookie __unused) {
143       ALOGI("%s", __FUNCTION__);
144       mParent.lastModelEvent = event;
145       mParent.monitor.notify();
146       return Void();
147     }
148   };
149 
TearDown()150   virtual void TearDown() override {}
151 
152   Monitor monitor;
153   // updated by soundModelCallback()
154   ISoundTriggerHwCallback::ModelEvent lastModelEvent;
155 
156  protected:
157   sp<ISoundTriggerHw> mSoundTriggerHal;
158   sp<SoundTriggerHwCallback> mCallback;
159 };
160 
161 /**
162  * Test ISoundTriggerHw::getProperties() method
163  *
164  * Verifies that:
165  *  - the implementation implements the method
166  *  - the method returns 0 (no error)
167  *  - the implementation supports at least one sound model and one key phrase
168  *  - the implementation supports at least VOICE_TRIGGER recognition mode
169  */
TEST_F(SoundTriggerHidlTest,GetProperties)170 TEST_F(SoundTriggerHidlTest, GetProperties) {
171   ISoundTriggerHw::Properties halProperties;
172   Return<void> hidlReturn;
173   int ret = -ENODEV;
174 
175   hidlReturn = mSoundTriggerHal->getProperties([&](int rc, auto res) {
176       ret = rc;
177       halProperties = res;
178   });
179 
180   EXPECT_TRUE(hidlReturn.isOk());
181   EXPECT_EQ(0, ret);
182   EXPECT_GT(halProperties.maxSoundModels, 0u);
183   EXPECT_GT(halProperties.maxKeyPhrases, 0u);
184   EXPECT_NE(0u, (halProperties.recognitionModes & (uint32_t)RecognitionMode::VOICE_TRIGGER));
185 }
186 
187 /**
188  * Test ISoundTriggerHw::loadPhraseSoundModel() method
189  *
190  * Verifies that:
191  *  - the implementation implements the method
192  *  - the implementation returns an error when passed a malformed sound model
193  *
194  * There is no way to verify that implementation actually can load a sound model because each
195  * sound model is vendor specific.
196  */
TEST_F(SoundTriggerHidlTest,LoadInvalidModelFail)197 TEST_F(SoundTriggerHidlTest, LoadInvalidModelFail) {
198   Return<void> hidlReturn;
199   int ret = -ENODEV;
200   ISoundTriggerHw::PhraseSoundModel model;
201   SoundModelHandle handle;
202 
203   model.common.type = SoundModelType::UNKNOWN;
204 
205   hidlReturn = mSoundTriggerHal->loadPhraseSoundModel(
206           model,
207           mCallback, 0, [&](int32_t retval, auto res) {
208       ret = retval;
209       handle = res;
210   });
211 
212   EXPECT_TRUE(hidlReturn.isOk());
213   EXPECT_NE(0, ret);
214   EXPECT_FALSE(monitor.wait(SHORT_TIMEOUT_PERIOD));
215 }
216 
217 /**
218  * Test ISoundTriggerHw::loadSoundModel() method
219  *
220  * Verifies that:
221  *  - the implementation returns error when passed a sound model with random data.
222  */
TEST_F(SoundTriggerHidlTest,LoadGenericSoundModelFail)223 TEST_F(SoundTriggerHidlTest, LoadGenericSoundModelFail) {
224   int ret = -ENODEV;
225   ISoundTriggerHw::SoundModel model;
226   SoundModelHandle handle = 0;
227 
228   model.type = SoundModelType::GENERIC;
229   model.data.resize(100);
230   for (auto& d : model.data) {
231     d = rand();
232   }
233 
234   Return<void> loadReturn = mSoundTriggerHal->loadSoundModel(
235       model,
236       mCallback, 0, [&](int32_t retval, auto res) {
237     ret = retval;
238     handle = res;
239   });
240 
241   EXPECT_TRUE(loadReturn.isOk());
242   EXPECT_NE(0, ret);
243   EXPECT_FALSE(monitor.wait(SHORT_TIMEOUT_PERIOD));
244 }
245 
246 /**
247  * Test ISoundTriggerHw::unloadSoundModel() method
248  *
249  * Verifies that:
250  *  - the implementation implements the method
251  *  - the implementation returns an error when called without a valid loaded sound model
252  *
253  */
TEST_F(SoundTriggerHidlTest,UnloadModelNoModelFail)254 TEST_F(SoundTriggerHidlTest, UnloadModelNoModelFail) {
255   Return<int32_t> hidlReturn(0);
256   SoundModelHandle halHandle = 0;
257 
258   hidlReturn = mSoundTriggerHal->unloadSoundModel(halHandle);
259 
260   EXPECT_TRUE(hidlReturn.isOk());
261   EXPECT_NE(0, hidlReturn);
262 }
263 
264 /**
265  * Test ISoundTriggerHw::startRecognition() method
266  *
267  * Verifies that:
268  *  - the implementation implements the method
269  *  - the implementation returns an error when called without a valid loaded sound model
270  *
271  * There is no way to verify that implementation actually starts recognition because no model can
272  * be loaded.
273  */
TEST_F(SoundTriggerHidlTest,StartRecognitionNoModelFail)274 TEST_F(SoundTriggerHidlTest, StartRecognitionNoModelFail) {
275     Return<int32_t> hidlReturn(0);
276     SoundModelHandle handle = 0;
277     PhraseRecognitionExtra phrase;
278     ISoundTriggerHw::RecognitionConfig config;
279 
280     config.captureHandle = 0;
281     config.captureDevice = AudioDevice::IN_BUILTIN_MIC;
282     phrase.id = 0;
283     phrase.recognitionModes = (uint32_t)RecognitionMode::VOICE_TRIGGER;
284     phrase.confidenceLevel = 0;
285 
286     config.phrases.setToExternal(&phrase, 1);
287 
288     hidlReturn = mSoundTriggerHal->startRecognition(handle, config, mCallback, 0);
289 
290     EXPECT_TRUE(hidlReturn.isOk());
291     EXPECT_NE(0, hidlReturn);
292 }
293 
294 /**
295  * Test ISoundTriggerHw::stopRecognition() method
296  *
297  * Verifies that:
298  *  - the implementation implements the method
299  *  - the implementation returns an error when called without an active recognition running
300  *
301  */
TEST_F(SoundTriggerHidlTest,StopRecognitionNoAStartFail)302 TEST_F(SoundTriggerHidlTest, StopRecognitionNoAStartFail) {
303     Return<int32_t> hidlReturn(0);
304     SoundModelHandle handle = 0;
305 
306     hidlReturn = mSoundTriggerHal->stopRecognition(handle);
307 
308     EXPECT_TRUE(hidlReturn.isOk());
309     EXPECT_NE(0, hidlReturn);
310 }
311 
312 /**
313  * Test ISoundTriggerHw::stopAllRecognitions() method
314  *
315  * Verifies that:
316  *  - the implementation implements this optional method or indicates it is not support by
317  *  returning -ENOSYS
318  */
TEST_F(SoundTriggerHidlTest,stopAllRecognitions)319 TEST_F(SoundTriggerHidlTest, stopAllRecognitions) {
320     Return<int32_t> hidlReturn(0);
321 
322     hidlReturn = mSoundTriggerHal->stopAllRecognitions();
323 
324     EXPECT_TRUE(hidlReturn.isOk());
325     EXPECT_TRUE(hidlReturn == 0 || hidlReturn == -ENOSYS);
326 }
327 
main(int argc,char ** argv)328 int main(int argc, char** argv) {
329     ::testing::AddGlobalTestEnvironment(SoundTriggerHidlEnvironment::Instance());
330     ::testing::InitGoogleTest(&argc, argv);
331     SoundTriggerHidlEnvironment::Instance()->init(&argc, argv);
332     int status = RUN_ALL_TESTS();
333     ALOGI("Test result = %d", status);
334     return status;
335 }
336