デコードされた音声を再生すると、ゴボゴボという音から金切り声、悪魔のような詠唱まで、さまざまな音を出すことができました。最も近い音は、早送り再生に似ており、再生は約 15 秒しか続きません。デコードおよび AudioSystem API メソッドのパラメーターを多数組み合わせて試してみましたが、何も機能していないようです。
では、このオーディオの歪みの原因は何ですか?
このファイルの Opusinfo には次のように表示されます。
Processing file "test.opus"...
New logical stream (#1, serial: 00002c88): type opus
Encoded with libopus 1.1
User comments section follows...
ENCODER=opusenc from opus-tools 0.1.9
Opus stream 1:
Pre-skip: 356
Playback gain: 0 dB
Channels: 1
Original sample rate: 44100Hz
Packet duration: 20.0ms (max), 20.0ms (avg), 20.0ms (min)
Page duration: 1000.0ms (max), 996.8ms (avg), 200.0ms (min)
Total data length: 1930655 bytes (overhead: 1.04%)
Playback length: 4m:09.173s
Average bitrate: 61.99 kb/s, w/o overhead: 61.34 kb/s
Logical stream 1 ended
このファイルは、VLC を使用して正しく再生されます。
ファイルをデコードするには、次のライブラリを使用しようとしています:
VorbisJava ( https://github.com/Gagravarr/VorbisJava/ ) - OGG コンテナーからフレームをプルするには
LibJitsi ( https://jitsi.org/Projects/LibJitsi ) - opus フレームのデコードに使用される、Opus 用の JNI ラッパーを備えています。
以下のSSCCE
package me.justinb.mediapad.audio;
import org.gagravarr.ogg.OggFile;
import org.gagravarr.ogg.OggPacket;
import org.jitsi.impl.neomedia.codec.audio.opus.Opus;
import javax.sound.sampled.*;
import java.io.*;
import java.nio.ByteBuffer;
public class OpusAudioPlayer {
private static int BUFFER_SIZE = 1024 * 1024;
private static int INPUT_BITRATE = 48000;
private static int OUTPUT_BITRATE = 44100;
private OggFile oggFile;
private long opusState;
private ByteBuffer decodeBuffer = ByteBuffer.allocate(BUFFER_SIZE);
private AudioFormat audioFormat = new AudioFormat(OUTPUT_BITRATE, 16, 1, true, false);
public static void main(String[] args) {
try {
OpusAudioPlayer opusAudioPlayer = new OpusAudioPlayer(new File("test.opus"));
opusAudioPlayer.play();
} catch (IOException e) {
e.printStackTrace();
}
}
public OpusAudioPlayer(File audioFile) throws IOException {
oggFile = new OggFile(new FileInputStream(audioFile));
opusState = Opus.decoder_create(INPUT_BITRATE, 1);
System.out.println("Audio format: " + audioFormat);
}
private byte[] decode(byte[] packetData) {
int frameSize = Opus.decoder_get_nb_samples(opusState, packetData, 0, packetData.length);
int decodedSamples = Opus.decode(opusState, packetData, 0, packetData.length, decodeBuffer.array(), 0, frameSize, 0);
if (decodedSamples < 0) {
System.out.println("Decode error: " + decodedSamples);
decodeBuffer.clear();
return null;
}
decodeBuffer.position(decodedSamples * 2); // 2 bytes per sample
decodeBuffer.flip();
byte[] decodedData = new byte[decodeBuffer.remaining()];
decodeBuffer.get(decodedData);
decodeBuffer.flip();
System.out.println(String.format("Encoded frame size: %d bytes", packetData.length));
System.out.println(String.format("Decoded frame size: %d bytes", decodedData.length));
System.out.println(String.format("Decoded %d samples", decodedSamples));
return decodedData;
}
public void play() {
int totalDecodedBytes = 0;
try {
SourceDataLine speaker = AudioSystem.getSourceDataLine(audioFormat);
OggPacket nextPacket = oggFile.getPacketReader().getNextPacket();
// Move to beginning of stream
while ( !nextPacket.isBeginningOfStream()) {
nextPacket = oggFile.getPacketReader().getNextPacket();
}
speaker.open();
speaker.start();
while(nextPacket != null) {
// Decode each packet
byte[] decodedData = decode(nextPacket.getData());
if(decodedData != null) {
// Write packet to SourceDataLine
speaker.write(decodedData, 0, decodedData.length);
totalDecodedBytes += decodedData.length;
}
nextPacket = oggFile.getPacketReader().getNextPacket();
}
speaker.drain();
speaker.close();
System.out.println(String.format("Decoded to %d bytes", totalDecodedBytes));
} catch (Exception e) {
e.printStackTrace();
}
}
}