Не удается исправить неверную среднеквадратичную шкалу

Моя команда и я создаем приложение, которое включает обойденный источник звука, отображающий значение rmsDb. Мы надеялись получить шкалу от -155 до 0, но значения, которые мы получаем, варьируются от -70 до -19. Мы не знаем, что не так с масштабом, так как мы все довольно плохо знакомы с аудио-программированием.

РЕДАКТИРОВАТЬ: Это относится к Google Project Tango, потому что используемое нами устройство - Lenovo Phab2Pro, которое является единственным коммерчески доступным устройством с поддержкой Tango, которое мы купили специально для работы со звуком в Tango. Таким образом, это влияет на принятие технологии танго.

public class RecordingThread {
private static final String LOG_TAG = RecordingThread.class.getSimpleName();
private static final int SAMPLE_RATE = 22050;
public float centerFrequency = 1000;
private short[] audioBuffer;
private short[] audioBuffer2;
private double rms;
private double mGain;
private double rmsdB;

public RecordingThread(AudioDataReceivedListener listener) {
    mListener = listener;
}

private boolean mShouldContinue;
private AudioDataReceivedListener mListener;
private Thread mThread;

public boolean recording() {
    return mThread != null;
}

public void startRecording() {
    if (mThread != null)
        return;

    mShouldContinue = true;
    mThread = new Thread(new Runnable() {
        @Override
        public void run() {
            record();
        }
    });
    mThread.start();
}

public void stopRecording() {
    if (mThread == null)
        return;

    mShouldContinue = false;
    mThread = null;
}

private void record() {
    rms = 0;
    android.os.Process.setThreadPriority(Process.THREAD_PRIORITY_URGENT_AUDIO);

    // buffer size in bytes
    int bufferSize = AudioRecord.getMinBufferSize(SAMPLE_RATE,
            AudioFormat.CHANNEL_IN_MONO,
            AudioFormat.ENCODING_PCM_16BIT);

    if (bufferSize == AudioRecord.ERROR || bufferSize == AudioRecord.ERROR_BAD_VALUE) {
        bufferSize = SAMPLE_RATE * 2;
    }

    audioBuffer = new short[bufferSize / 2];
    audioBuffer2 = new short[bufferSize / 2];

    AudioRecord record = new AudioRecord(MediaRecorder.AudioSource.DEFAULT,
            SAMPLE_RATE,
            AudioFormat.CHANNEL_IN_MONO,
            AudioFormat.ENCODING_PCM_16BIT,
            bufferSize);

    if (record.getState() != AudioRecord.STATE_INITIALIZED) {
        Log.e(LOG_TAG, "Audio Record can't initialize!");
        return;
    }
    record.startRecording();

    Log.v(LOG_TAG, "Start recording");

    long shortsRead = 0;
    while (mShouldContinue) {
        int numberOfShort = record.read(audioBuffer, 0, audioBuffer.length);
        shortsRead += numberOfShort;

        /*
         * Noise level meter begins here
         */
        // Compute the RMS value. (Note that this does not remove DC).
        for (int i = 2; i < audioBuffer.length; i++) {
            if(i % 3 == 0) {
                audioBuffer2[i] = BiQuad.bqfilter(audioBuffer[i], audioBuffer[i - 1], audioBuffer[i - 2], audioBuffer2[i], audioBuffer2[i - 1], audioBuffer2[i - 2], SAMPLE_RATE, centerFrequency, 5);
                rms += audioBuffer2[i] * audioBuffer2[i];
            }
        }
        rms = Math.sqrt(rms / audioBuffer2.length);
        mGain = 1.0/32767; //0.0044;
        rmsdB = 20.0 * Math.log10(mGain * rms);
    }

    record.stop();
    record.release();

    Log.v(LOG_TAG, String.format("Recording stopped. Samples read: %d", shortsRead));
    }
}

Любая помощь будет принята с благодарностью.

1 ответ

Некоторая калибровка, вероятно, необходима, то есть добавление константы к вашему значению rmsdB.

Сделайте несколько измерений и сравните со стандартным измерителем уровня звука.

Составляя график интенсивности звука (в дБ, по оси Y) в зависимости от вашего значения rmsdB (по оси X), вы можете подогнать линию к этим данным, и тогда пересечение Y графика покажет Вы константу, чтобы добавить, например,

rmsdB = calibrationCostant + 20.0 * Math.log10(mGain * rms);
Другие вопросы по тегам