Move opus decoding code

This commit is contained in:
Iwan Timmer
2015-03-09 12:16:57 +01:00
parent be584fb203
commit 40508c2ac2
8 changed files with 114 additions and 96 deletions

View File

@@ -5,14 +5,14 @@ import java.net.UnknownHostException;
import java.io.File;
import java.io.IOException;
import com.limelight.binding.audio.AlsaAudioRenderer;
import com.limelight.binding.audio.AlsaAudioDecoderRenderer;
import com.limelight.binding.video.ImxDecoder;
import com.limelight.binding.video.ImxDecoderRenderer;
import com.limelight.binding.crypto.PcCryptoProvider;
import com.limelight.binding.video.OmxDecoder;
import com.limelight.binding.video.OmxDecoderRenderer;
import com.limelight.binding.video.AbstractVideoRenderer;
import com.limelight.nvstream.av.audio.AudioRenderer;
import com.limelight.nvstream.av.audio.AudioDecoderRenderer;
import com.limelight.nvstream.av.video.VideoDecoderRenderer;
import com.limelight.nvstream.http.LimelightCryptoProvider;
import com.limelight.LimeLog;
@@ -55,7 +55,7 @@ public class PlatformBinding {
* Gets an instance of an audio decoder/renderer.
* @return an audio decoder and renderer
*/
public static AudioRenderer getAudioRenderer(String device) {
public static AudioDecoderRenderer getAudioRenderer(String device) {
//Try to load local libopus
try {
Runtime.getRuntime().load(new File(".").getCanonicalPath()+File.separator+"libopus.so");
@@ -66,7 +66,7 @@ public class PlatformBinding {
//Use system opus library
}
return new AlsaAudioRenderer(device);
return new AlsaAudioDecoderRenderer(device);
}
/**

View File

@@ -0,0 +1,53 @@
package com.limelight.binding.audio;
import com.limelight.LimeLog;
import com.limelight.nvstream.av.audio.AudioDecoderRenderer;
/**
* Audio renderer implementation
* @author Iwan Timmer
*/
public class AlsaAudioDecoderRenderer implements AudioDecoderRenderer {
private String device;
private byte[] decodedData;
public AlsaAudioDecoderRenderer(String device) {
this.device = device;
this.decodedData = new byte[OpusDecoder.getMaxOutputShorts()*2];
int err;
err = OpusDecoder.init();
if (err != 0) {
throw new IllegalStateException("Opus decoder failed to initialize");
}
}
@Override
public boolean streamInitialize() {
return AlsaAudio.init(OpusDecoder.getChannelCount(), OpusDecoder.getSampleRate(), device) == 0;
}
@Override
public void playAudio(byte[] bytes, int offset, int length) {
int decodeLen = OpusDecoder.decode(bytes, offset, length, decodedData);
if (decodeLen > 0) {
//Value of decode is frames (shorts) decoded per channel
decodeLen *= 2*OpusDecoder.getChannelCount();
int rc = AlsaAudio.play(decodedData, 0, decodeLen);
if (rc<0)
LimeLog.warning("Alsa error from writei: "+rc);
else if (rc!=length/4)
LimeLog.warning("Alsa short write, write "+rc+" frames");
}
}
@Override
public void streamClosing() {
AlsaAudio.close();
}
}

View File

@@ -1,43 +0,0 @@
package com.limelight.binding.audio;
import com.limelight.LimeLog;
import com.limelight.nvstream.av.audio.AudioRenderer;
/**
* Audio renderer implementation
* @author Iwan Timmer
*/
public class AlsaAudioRenderer implements AudioRenderer {
private String device;
public AlsaAudioRenderer(String device) {
this.device = device;
}
@Override
public boolean streamInitialized(int channelCount, int sampleRate) {
return AlsaAudio.init(channelCount, sampleRate, device) == 0;
}
@Override
public void playDecodedAudio(byte[] bytes, int offset, int length) {
int rc = AlsaAudio.play(bytes, offset, length);
if (rc<0)
LimeLog.warning("Alsa error from writei: "+rc);
else if (rc!=length/4)
LimeLog.warning("Alsa short write, write "+rc+" frames");
}
@Override
public void streamClosing() {
AlsaAudio.close();
}
@Override
public int getCapabilities() {
return 0;
}
}

View File

@@ -0,0 +1,35 @@
package com.limelight.binding.audio;
import com.limelight.nvstream.av.audio.AudioDecoderRenderer;
/**
* Fake implementation for audio renderer.
* @author Iwan Timmer
*/
public class FakeAudioDecoderRenderer implements AudioDecoderRenderer {
private int dataSize;
private long last;
@Override
public boolean streamInitialize() {
System.out.println("Fake audio output");
return true;
}
@Override
public void playAudio(byte[] audioData, int offset, int length) {
if (System.currentTimeMillis()>last+2000) {
int bitrate = (dataSize/2)/1024;
System.out.println("Audio " + bitrate + "kB/s");
dataSize = 0;
last = System.currentTimeMillis();
}
dataSize += length;
}
@Override
public void streamClosing() {
}
}

View File

@@ -1,41 +0,0 @@
package com.limelight.binding.audio;
import com.limelight.nvstream.av.audio.AudioRenderer;
/**
* Fake implementation for audio renderer.
* @author Iwan Timmer
*/
public class FakeAudioRenderer implements AudioRenderer {
private int dataSize;
private long last;
@Override
public boolean streamInitialized(int channelCount, int sampleRate) {
System.out.println("Fake " + channelCount + " channel " + sampleRate + " samplerate audio output");
last = System.currentTimeMillis();
return true;
}
@Override
public void playDecodedAudio(byte[] audioData, int offset, int length) {
if (System.currentTimeMillis()>last+2000) {
int bitrate = (dataSize/2)/1024;
System.out.println("Audio " + bitrate + "kB/s");
dataSize = 0;
last = System.currentTimeMillis();
}
dataSize += length;
}
@Override
public void streamClosing() {
}
@Override
public int getCapabilities() {
return CAPABILITY_DIRECT_SUBMIT;
}
}

View File

@@ -0,0 +1,14 @@
package com.limelight.binding.audio;
public class OpusDecoder {
static {
System.loadLibrary("nv_opus_dec");
}
public static native int init();
public static native void destroy();
public static native int getChannelCount();
public static native int getMaxOutputShorts();
public static native int getSampleRate();
public static native int decode(byte[] indata, int inoff, int inlen, byte[] outpcmdata);
}