Post OfflineLicenseHelper interactions to its internal handler thread

`DefaultDrmSession(Manager)` expect most of their methods to be called
on the 'playback thread'. There isn't a playback thread in the case of
`OfflineLicenseHelper`, but in that case it's the thread backing
`DefaultDrmSessionManager.playbackLooper`, which is `OfflineLicenseHelper.handlerThread`.

PiperOrigin-RevId: 520053006
This commit is contained in:
ibaker 2023-03-28 16:39:42 +00:00 committed by Tianyi Feng
parent f88280dc78
commit 376bddef47

View File

@ -19,6 +19,7 @@ import android.media.MediaDrm;
import android.os.ConditionVariable; import android.os.ConditionVariable;
import android.os.Handler; import android.os.Handler;
import android.os.HandlerThread; import android.os.HandlerThread;
import android.os.Looper;
import android.util.Pair; import android.util.Pair;
import androidx.annotation.Nullable; import androidx.annotation.Nullable;
import androidx.annotation.RequiresApi; import androidx.annotation.RequiresApi;
@ -31,8 +32,11 @@ import androidx.media3.exoplayer.analytics.PlayerId;
import androidx.media3.exoplayer.drm.DefaultDrmSessionManager.Mode; import androidx.media3.exoplayer.drm.DefaultDrmSessionManager.Mode;
import androidx.media3.exoplayer.drm.DrmSession.DrmSessionException; import androidx.media3.exoplayer.drm.DrmSession.DrmSessionException;
import androidx.media3.exoplayer.source.MediaSource.MediaPeriodId; import androidx.media3.exoplayer.source.MediaSource.MediaPeriodId;
import com.google.common.util.concurrent.SettableFuture;
import java.util.Map; import java.util.Map;
import java.util.UUID; import java.util.UUID;
import java.util.concurrent.ExecutionException;
import org.checkerframework.checker.nullness.compatqual.NullableType;
/** Helper class to download, renew and release offline licenses. */ /** Helper class to download, renew and release offline licenses. */
@RequiresApi(18) @RequiresApi(18)
@ -42,9 +46,10 @@ public final class OfflineLicenseHelper {
private static final Format FORMAT_WITH_EMPTY_DRM_INIT_DATA = private static final Format FORMAT_WITH_EMPTY_DRM_INIT_DATA =
new Format.Builder().setDrmInitData(new DrmInitData()).build(); new Format.Builder().setDrmInitData(new DrmInitData()).build();
private final ConditionVariable conditionVariable; private final ConditionVariable drmListenerConditionVariable;
private final DefaultDrmSessionManager drmSessionManager; private final DefaultDrmSessionManager drmSessionManager;
private final HandlerThread handlerThread; private final HandlerThread handlerThread;
private final Handler handler;
private final DrmSessionEventListener.EventDispatcher eventDispatcher; private final DrmSessionEventListener.EventDispatcher eventDispatcher;
/** /**
@ -156,28 +161,29 @@ public final class OfflineLicenseHelper {
this.eventDispatcher = eventDispatcher; this.eventDispatcher = eventDispatcher;
handlerThread = new HandlerThread("ExoPlayer:OfflineLicenseHelper"); handlerThread = new HandlerThread("ExoPlayer:OfflineLicenseHelper");
handlerThread.start(); handlerThread.start();
conditionVariable = new ConditionVariable(); handler = new Handler(handlerThread.getLooper());
drmListenerConditionVariable = new ConditionVariable();
DrmSessionEventListener eventListener = DrmSessionEventListener eventListener =
new DrmSessionEventListener() { new DrmSessionEventListener() {
@Override @Override
public void onDrmKeysLoaded(int windowIndex, @Nullable MediaPeriodId mediaPeriodId) { public void onDrmKeysLoaded(int windowIndex, @Nullable MediaPeriodId mediaPeriodId) {
conditionVariable.open(); drmListenerConditionVariable.open();
} }
@Override @Override
public void onDrmSessionManagerError( public void onDrmSessionManagerError(
int windowIndex, @Nullable MediaPeriodId mediaPeriodId, Exception e) { int windowIndex, @Nullable MediaPeriodId mediaPeriodId, Exception e) {
conditionVariable.open(); drmListenerConditionVariable.open();
} }
@Override @Override
public void onDrmKeysRestored(int windowIndex, @Nullable MediaPeriodId mediaPeriodId) { public void onDrmKeysRestored(int windowIndex, @Nullable MediaPeriodId mediaPeriodId) {
conditionVariable.open(); drmListenerConditionVariable.open();
} }
@Override @Override
public void onDrmKeysRemoved(int windowIndex, @Nullable MediaPeriodId mediaPeriodId) { public void onDrmKeysRemoved(int windowIndex, @Nullable MediaPeriodId mediaPeriodId) {
conditionVariable.open(); drmListenerConditionVariable.open();
} }
}; };
eventDispatcher.addEventListener(new Handler(handlerThread.getLooper()), eventListener); eventDispatcher.addEventListener(new Handler(handlerThread.getLooper()), eventListener);
@ -193,7 +199,8 @@ public final class OfflineLicenseHelper {
*/ */
public synchronized byte[] downloadLicense(Format format) throws DrmSessionException { public synchronized byte[] downloadLicense(Format format) throws DrmSessionException {
Assertions.checkArgument(format.drmInitData != null); Assertions.checkArgument(format.drmInitData != null);
return blockingKeyRequest(DefaultDrmSessionManager.MODE_DOWNLOAD, null, format); return acquireSessionAndGetOfflineLicenseKeySetIdOnHandlerThread(
DefaultDrmSessionManager.MODE_DOWNLOAD, /* offlineLicenseKeySetId= */ null, format);
} }
/** /**
@ -206,7 +213,7 @@ public final class OfflineLicenseHelper {
public synchronized byte[] renewLicense(byte[] offlineLicenseKeySetId) public synchronized byte[] renewLicense(byte[] offlineLicenseKeySetId)
throws DrmSessionException { throws DrmSessionException {
Assertions.checkNotNull(offlineLicenseKeySetId); Assertions.checkNotNull(offlineLicenseKeySetId);
return blockingKeyRequest( return acquireSessionAndGetOfflineLicenseKeySetIdOnHandlerThread(
DefaultDrmSessionManager.MODE_DOWNLOAD, DefaultDrmSessionManager.MODE_DOWNLOAD,
offlineLicenseKeySetId, offlineLicenseKeySetId,
FORMAT_WITH_EMPTY_DRM_INIT_DATA); FORMAT_WITH_EMPTY_DRM_INIT_DATA);
@ -221,7 +228,7 @@ public final class OfflineLicenseHelper {
public synchronized void releaseLicense(byte[] offlineLicenseKeySetId) public synchronized void releaseLicense(byte[] offlineLicenseKeySetId)
throws DrmSessionException { throws DrmSessionException {
Assertions.checkNotNull(offlineLicenseKeySetId); Assertions.checkNotNull(offlineLicenseKeySetId);
blockingKeyRequest( acquireSessionAndGetOfflineLicenseKeySetIdOnHandlerThread(
DefaultDrmSessionManager.MODE_RELEASE, DefaultDrmSessionManager.MODE_RELEASE,
offlineLicenseKeySetId, offlineLicenseKeySetId,
FORMAT_WITH_EMPTY_DRM_INIT_DATA); FORMAT_WITH_EMPTY_DRM_INIT_DATA);
@ -237,25 +244,39 @@ public final class OfflineLicenseHelper {
public synchronized Pair<Long, Long> getLicenseDurationRemainingSec(byte[] offlineLicenseKeySetId) public synchronized Pair<Long, Long> getLicenseDurationRemainingSec(byte[] offlineLicenseKeySetId)
throws DrmSessionException { throws DrmSessionException {
Assertions.checkNotNull(offlineLicenseKeySetId); Assertions.checkNotNull(offlineLicenseKeySetId);
drmSessionManager.setPlayer(handlerThread.getLooper(), PlayerId.UNSET); DrmSession drmSession;
drmSessionManager.prepare(); try {
DrmSession drmSession = drmSession =
openBlockingKeyRequest( acquireFirstSessionOnHandlerThread(
DefaultDrmSessionManager.MODE_QUERY, DefaultDrmSessionManager.MODE_QUERY,
offlineLicenseKeySetId, offlineLicenseKeySetId,
FORMAT_WITH_EMPTY_DRM_INIT_DATA); FORMAT_WITH_EMPTY_DRM_INIT_DATA);
DrmSessionException error = drmSession.getError(); } catch (DrmSessionException e) {
Pair<Long, Long> licenseDurationRemainingSec = if (e.getCause() instanceof KeysExpiredException) {
WidevineUtil.getLicenseDurationRemainingSec(drmSession);
drmSession.release(eventDispatcher);
drmSessionManager.release();
if (error != null) {
if (error.getCause() instanceof KeysExpiredException) {
return Pair.create(0L, 0L); return Pair.create(0L, 0L);
} }
throw error; throw e;
}
SettableFuture<Pair<Long, Long>> licenseDurationRemainingSec = SettableFuture.create();
handler.post(
() -> {
try {
licenseDurationRemainingSec.set(
Assertions.checkNotNull(WidevineUtil.getLicenseDurationRemainingSec(drmSession)));
} catch (Throwable e) {
licenseDurationRemainingSec.setException(e);
} finally {
drmSession.release(eventDispatcher);
}
});
try {
return licenseDurationRemainingSec.get();
} catch (ExecutionException | InterruptedException e) {
throw new IllegalStateException(e);
} finally {
releaseManagerOnHandlerThread();
} }
return Assertions.checkNotNull(licenseDurationRemainingSec);
} }
/** Releases the helper. Should be called when the helper is no longer required. */ /** Releases the helper. Should be called when the helper is no longer required. */
@ -263,30 +284,146 @@ public final class OfflineLicenseHelper {
handlerThread.quit(); handlerThread.quit();
} }
private byte[] blockingKeyRequest( /**
* Returns the result of {@link DrmSession#getOfflineLicenseKeySetId()}, or throws {@link
* NullPointerException} if it's null.
*
* <p>This method takes care of acquiring and releasing the {@link DrmSessionManager} and {@link
* DrmSession} instances needed.
*/
private byte[] acquireSessionAndGetOfflineLicenseKeySetIdOnHandlerThread(
@Mode int licenseMode, @Nullable byte[] offlineLicenseKeySetId, Format format) @Mode int licenseMode, @Nullable byte[] offlineLicenseKeySetId, Format format)
throws DrmSessionException { throws DrmSessionException {
drmSessionManager.setPlayer(handlerThread.getLooper(), PlayerId.UNSET); DrmSession drmSession =
drmSessionManager.prepare(); acquireFirstSessionOnHandlerThread(licenseMode, offlineLicenseKeySetId, format);
DrmSession drmSession = openBlockingKeyRequest(licenseMode, offlineLicenseKeySetId, format);
DrmSessionException error = drmSession.getError(); SettableFuture<byte @NullableType []> keySetId = SettableFuture.create();
byte[] keySetId = drmSession.getOfflineLicenseKeySetId(); handler.post(
() -> {
try {
keySetId.set(drmSession.getOfflineLicenseKeySetId());
} catch (Throwable e) {
keySetId.setException(e);
} finally {
drmSession.release(eventDispatcher); drmSession.release(eventDispatcher);
drmSessionManager.release();
if (error != null) {
throw error;
} }
return Assertions.checkNotNull(keySetId); });
try {
return Assertions.checkNotNull(keySetId.get());
} catch (ExecutionException | InterruptedException e) {
throw new IllegalStateException(e);
} finally {
releaseManagerOnHandlerThread();
}
} }
private DrmSession openBlockingKeyRequest( /**
@Mode int licenseMode, @Nullable byte[] offlineLicenseKeySetId, Format format) { * Calls {@link DrmSessionManager#acquireSession(DrmSessionEventListener.EventDispatcher, Format)}
* on {@link #handlerThread} and blocks until a callback is received via {@link
* DrmSessionEventListener}.
*
* <p>If key loading failed and {@link DrmSession#getState()} returns {@link
* DrmSession#STATE_ERROR} then this method releases the session and throws {@link
* DrmSession#getError()}.
*
* <p>Callers are responsible for the following:
*
* <ul>
* <li>Ensuring the {@link
* DrmSessionManager#acquireSession(DrmSessionEventListener.EventDispatcher, Format)} call
* will trigger a callback to {@link DrmSessionEventListener} (e.g. it will load new keys).
* If not, this method will block forever.
* <li>Releasing the returned {@link DrmSession} instance (on {@link #handlerThread}).
* <li>Releasing {@link #drmSessionManager} if a {@link DrmSession} instance is returned (the
* manager will be released before an exception is thrown).
* </ul>
*/
private DrmSession acquireFirstSessionOnHandlerThread(
@Mode int licenseMode, @Nullable byte[] offlineLicenseKeySetId, Format format)
throws DrmSessionException {
Assertions.checkNotNull(format.drmInitData); Assertions.checkNotNull(format.drmInitData);
SettableFuture<DrmSession> drmSessionFuture = SettableFuture.create();
drmListenerConditionVariable.close();
handler.post(
() -> {
try {
drmSessionManager.setPlayer(Assertions.checkNotNull(Looper.myLooper()), PlayerId.UNSET);
drmSessionManager.prepare();
try {
drmSessionManager.setMode(licenseMode, offlineLicenseKeySetId); drmSessionManager.setMode(licenseMode, offlineLicenseKeySetId);
conditionVariable.close(); drmSessionFuture.set(
DrmSession drmSession = drmSessionManager.acquireSession(eventDispatcher, format); Assertions.checkNotNull(
// Block current thread until key loading is finished drmSessionManager.acquireSession(eventDispatcher, format)));
conditionVariable.block(); } catch (Throwable e) {
return Assertions.checkNotNull(drmSession); drmSessionManager.release();
throw e;
}
} catch (Throwable e) {
drmSessionFuture.setException(e);
}
});
DrmSession drmSession;
try {
drmSession = drmSessionFuture.get();
} catch (ExecutionException | InterruptedException e) {
throw new IllegalStateException(e);
}
// drmListenerConditionVariable will be opened by a callback to this.eventDispatcher when key
// loading is complete (drmSession.state == STATE_OPENED_WITH_KEYS) or has failed
// (drmSession.state == STATE_ERROR).
drmListenerConditionVariable.block();
SettableFuture<@NullableType DrmSessionException> drmSessionErrorFuture =
SettableFuture.create();
handler.post(
() -> {
try {
DrmSessionException drmSessionError = drmSession.getError();
if (drmSession.getState() == DrmSession.STATE_ERROR) {
drmSession.release(eventDispatcher);
drmSessionManager.release();
}
drmSessionErrorFuture.set(drmSessionError);
} catch (Throwable e) {
drmSessionErrorFuture.setException(e);
drmSession.release(eventDispatcher);
drmSessionManager.release();
}
});
try {
DrmSessionException drmSessionError = drmSessionErrorFuture.get();
if (drmSessionError != null) {
throw drmSessionError;
} else {
return drmSession;
}
} catch (InterruptedException | ExecutionException e) {
throw new IllegalStateException(e);
}
}
/**
* Calls {@link DrmSessionManager#release()} on {@link #handlerThread} and blocks until it's
* complete.
*/
private void releaseManagerOnHandlerThread() {
SettableFuture<Void> result = SettableFuture.create();
handler.post(
() -> {
try {
drmSessionManager.release();
result.set(null);
} catch (Throwable e) {
result.setException(e);
}
});
try {
result.get();
} catch (InterruptedException | ExecutionException e) {
throw new IllegalStateException(e);
}
} }
} }