ex
Fork of mbed-os-example-mbed5-blinky by
Diff: dcs-sdk-java-master/app/src/main/java/com/baidu/duer/dcs/androidsystemimpl/wakeup/WakeUpDecodeThread.java
- Revision:
- 45:2aa9f933c8d2
--- /dev/null Thu Jan 01 00:00:00 1970 +0000
+++ b/dcs-sdk-java-master/app/src/main/java/com/baidu/duer/dcs/androidsystemimpl/wakeup/WakeUpDecodeThread.java Tue Jul 18 16:34:48 2017 +0800
@@ -0,0 +1,225 @@
+/*
+ * *
+ * Copyright (c) 2017 Baidu, Inc. All Rights Reserved.
+ *
+ * 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.
+ */
+package com.baidu.duer.dcs.androidsystemimpl.wakeup;
+
+import android.os.Handler;
+
+import com.baidu.duer.dcs.util.LogUtil;
+
+import java.util.concurrent.LinkedBlockingDeque;
+
+/**
+ * 唤醒声音处理-唤醒消费者
+ * <p>
+ * Created by guxiuzhong@baidu.com on 2017/6/22.
+ */
+public class WakeUpDecodeThread extends Thread {
+ private static final String TAG = WakeUpDecodeThread.class.getSimpleName();
+ // 唤醒成功
+ private static final int WAKEUP_SUCCEED = 1;
+ // 唤醒词位置
+ private int voiceOffset;
+ private boolean isWakeUp;
+ private WakeUpNative wakeUpNative;
+ private Handler handler;
+ private volatile boolean isStart = false;
+ private LinkedBlockingDeque<byte[]> linkedBlockingDeque;
+
+ public WakeUpDecodeThread(LinkedBlockingDeque<byte[]> linkedBlockingDeque,
+ WakeUpNative wakeUpNative,
+ Handler handler) {
+ this.linkedBlockingDeque = linkedBlockingDeque;
+ this.wakeUpNative = wakeUpNative;
+ this.handler = handler;
+ }
+
+ /**
+ * 开始唤醒
+ */
+ public void startWakeUp() {
+ if (isStart) {
+ return;
+ }
+ isStart = true;
+ this.start();
+ }
+
+ public boolean isStart() {
+ return isStart;
+ }
+
+ /**
+ * 停止唤醒
+ */
+ public void stopWakeUp() {
+ isStart = false;
+ }
+
+ @Override
+ public void run() {
+ super.run();
+ LogUtil.i(TAG, "wakeup wakeUpDecode start" );
+ while (isStart) {
+ try {
+ byte[] data = linkedBlockingDeque.take();
+
+ // 暂时不检测vad
+ // int volume = calculateVolume(data, 16);
+ // LogUtil.i(TAG, "wakeup audioRecord Volume:" + volume);
+ // if (volume <= 0) {
+ // continue;
+ // }
+
+ if (data.length > 0) {
+ // 是否为最后一帧数据
+ boolean isEnd = false;
+ short[] arr = byteArray2ShortArray(data, data.length / 2);
+ int ret = wakeUpNative.wakeUpDecode(
+ arr,
+ arr.length,
+ "",
+ 1,
+ -1,
+ true,
+ voiceOffset++,
+ isEnd
+ );
+ // 唤醒成功
+ if (ret == WAKEUP_SUCCEED) {
+ LogUtil.i(TAG, "wakeup wakeUpDecode ret:" + ret);
+ isWakeUp = true;
+ stopWakeUp();
+ LogUtil.i(TAG, "wakeup wakeUpDecode succeed !!");
+ break;
+ }
+ }
+ } catch (InterruptedException e) {
+ e.printStackTrace();
+ }
+ }
+ LogUtil.i(TAG, "wakeup after wakeUpDecode over !!");
+ LogUtil.i(TAG, "wakeup after linkedBlockingDeque size:" + linkedBlockingDeque.size());
+
+ if (isWakeUp) {
+ if (listener != null) {
+ handler.post(new Runnable() {
+ @Override
+ public void run() {
+ listener.onWakeUpSucceed();
+ }
+ });
+ }
+ }
+ // 重置
+ voiceOffset = 0;
+ isWakeUp = false;
+ }
+
+ /**
+ * 将byte数组转为short数组
+ *
+ * @param data byte数组的音频数据
+ * @param items short数组的大小
+ * @return 转换后的short数组
+ */
+ private short[] byteArray2ShortArray(byte[] data, int items) {
+ short[] retVal = new short[items];
+ for (int i = 0; i < retVal.length; i++) {
+ retVal[i] = (short) ((data[i * 2] & 0xff) | (data[i * 2 + 1] & 0xff) << 8);
+ }
+ return retVal;
+ }
+
+ /**
+ * 根据输入的音量大小来判断用户是否有语音输入
+ *
+ * @param var0 原始音频数据
+ * @param var1 音频数据格式位PCM,16位每个样本,比如如果音频数据格式设为AudioFormat.ENCODING_PCM_16BIT,则参数传16
+ * @return >0 代表有音频输入
+ */
+ private int calculateVolume(byte[] var0, int var1) {
+ int[] var3 = null;
+ int var4 = var0.length;
+ int var2;
+ if (var1 == 8) {
+ var3 = new int[var4];
+ for (var2 = 0; var2 < var4; ++var2) {
+ var3[var2] = var0[var2];
+ }
+ } else if (var1 == 16) {
+ var3 = new int[var4 / 2];
+ for (var2 = 0; var2 < var4 / 2; ++var2) {
+ byte var5 = var0[var2 * 2];
+ byte var6 = var0[var2 * 2 + 1];
+ int var13;
+ if (var5 < 0) {
+ var13 = var5 + 256;
+ } else {
+ var13 = var5;
+ }
+ short var7 = (short) (var13 + 0);
+ if (var6 < 0) {
+ var13 = var6 + 256;
+ } else {
+ var13 = var6;
+ }
+ var3[var2] = (short) (var7 + (var13 << 8));
+ }
+ }
+ int[] var8 = var3;
+ if (var3 != null && var3.length != 0) {
+ float var10 = 0.0F;
+ for (int var11 = 0; var11 < var8.length; ++var11) {
+ var10 += (float) (var8[var11] * var8[var11]);
+ }
+ var10 /= (float) var8.length;
+ float var12 = 0.0F;
+ for (var4 = 0; var4 < var8.length; ++var4) {
+ var12 += (float) var8[var4];
+ }
+ var12 /= (float) var8.length;
+ var4 = (int) (Math.pow(2.0D, (double) (var1 - 1)) - 1.0D);
+ double var14 = Math.sqrt((double) (var10 - var12 * var12));
+ int var9;
+ if ((var9 = (int) (10.0D * Math.log10(var14 * 10.0D * Math.sqrt(2.0D) / (double) var4 + 1.0D))) < 0) {
+ var9 = 0;
+ }
+ if (var9 > 10) {
+ var9 = 10;
+ }
+ return var9;
+ } else {
+ return 0;
+ }
+ }
+
+ private IWakeUpListener listener;
+
+ public void setWakeUpListener(IWakeUpListener listener) {
+ this.listener = listener;
+ }
+
+ /**
+ * 唤醒成功回调接口
+ */
+ public interface IWakeUpListener {
+ /**
+ * 唤醒成功后回调
+ */
+ void onWakeUpSucceed();
+ }
+}
\ No newline at end of file
