2017-06-16 09:23:50 -07:00
|
|
|
/*
|
|
|
|
|
* Copyright 2017 The WebRTC project authors. All Rights Reserved.
|
|
|
|
|
*
|
|
|
|
|
* Use of this source code is governed by a BSD-style license
|
|
|
|
|
* that can be found in the LICENSE file in the root of the source
|
|
|
|
|
* tree. An additional intellectual property rights grant can be found
|
|
|
|
|
* in the file PATENTS. All contributing project authors may
|
|
|
|
|
* be found in the AUTHORS file in the root of the source tree.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
package org.webrtc;
|
|
|
|
|
|
|
|
|
|
import java.nio.ByteBuffer;
|
|
|
|
|
import org.webrtc.VideoFrame.I420Buffer;
|
|
|
|
|
|
|
|
|
|
/** Implementation of an I420 VideoFrame buffer. */
|
|
|
|
|
class I420BufferImpl implements VideoFrame.I420Buffer {
|
|
|
|
|
private final int width;
|
|
|
|
|
private final int height;
|
2017-07-28 07:12:23 -07:00
|
|
|
private final ByteBuffer dataY;
|
|
|
|
|
private final ByteBuffer dataU;
|
|
|
|
|
private final ByteBuffer dataV;
|
2017-07-18 11:33:39 -07:00
|
|
|
private final int strideY;
|
|
|
|
|
private final int strideU;
|
|
|
|
|
private final int strideV;
|
2017-07-28 07:12:23 -07:00
|
|
|
private final Runnable releaseCallback;
|
2017-08-21 08:02:58 -07:00
|
|
|
private final Object refCountLock = new Object();
|
2017-06-16 09:23:50 -07:00
|
|
|
|
2017-07-18 11:33:39 -07:00
|
|
|
private int refCount;
|
|
|
|
|
|
2017-07-28 07:12:23 -07:00
|
|
|
/** Constructs an I420Buffer backed by existing data. */
|
|
|
|
|
I420BufferImpl(int width, int height, ByteBuffer dataY, int strideY, ByteBuffer dataU,
|
|
|
|
|
int strideU, ByteBuffer dataV, int strideV, Runnable releaseCallback) {
|
2017-06-16 09:23:50 -07:00
|
|
|
this.width = width;
|
|
|
|
|
this.height = height;
|
2017-07-28 07:12:23 -07:00
|
|
|
this.dataY = dataY;
|
|
|
|
|
this.dataU = dataU;
|
|
|
|
|
this.dataV = dataV;
|
2017-07-18 11:33:39 -07:00
|
|
|
this.strideY = strideY;
|
|
|
|
|
this.strideU = strideU;
|
|
|
|
|
this.strideV = strideV;
|
|
|
|
|
this.releaseCallback = releaseCallback;
|
|
|
|
|
|
|
|
|
|
this.refCount = 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** Allocates an empty I420Buffer suitable for an image of the given dimensions. */
|
|
|
|
|
static I420BufferImpl allocate(int width, int height) {
|
|
|
|
|
int chromaHeight = (height + 1) / 2;
|
|
|
|
|
int strideUV = (width + 1) / 2;
|
|
|
|
|
int yPos = 0;
|
|
|
|
|
int uPos = yPos + width * height;
|
|
|
|
|
int vPos = uPos + strideUV * chromaHeight;
|
2017-07-28 07:12:23 -07:00
|
|
|
|
2017-07-18 11:33:39 -07:00
|
|
|
ByteBuffer buffer = ByteBuffer.allocateDirect(width * height + 2 * strideUV * chromaHeight);
|
2017-07-28 07:12:23 -07:00
|
|
|
|
|
|
|
|
buffer.position(yPos);
|
|
|
|
|
buffer.limit(uPos);
|
|
|
|
|
ByteBuffer dataY = buffer.slice();
|
|
|
|
|
|
|
|
|
|
buffer.position(uPos);
|
|
|
|
|
buffer.limit(vPos);
|
|
|
|
|
ByteBuffer dataU = buffer.slice();
|
|
|
|
|
|
|
|
|
|
buffer.position(vPos);
|
|
|
|
|
buffer.limit(vPos + strideUV * chromaHeight);
|
|
|
|
|
ByteBuffer dataV = buffer.slice();
|
|
|
|
|
|
|
|
|
|
return new I420BufferImpl(width, height, dataY, width, dataU, strideUV, dataV, strideUV, null);
|
2017-06-16 09:23:50 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
public int getWidth() {
|
|
|
|
|
return width;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
public int getHeight() {
|
|
|
|
|
return height;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
public ByteBuffer getDataY() {
|
2017-09-29 11:33:52 -07:00
|
|
|
// Return a slice to prevent relative reads from changing the position.
|
|
|
|
|
return dataY.slice();
|
2017-06-16 09:23:50 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
public ByteBuffer getDataU() {
|
2017-09-29 11:33:52 -07:00
|
|
|
// Return a slice to prevent relative reads from changing the position.
|
|
|
|
|
return dataU.slice();
|
2017-06-16 09:23:50 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
public ByteBuffer getDataV() {
|
2017-09-29 11:33:52 -07:00
|
|
|
// Return a slice to prevent relative reads from changing the position.
|
|
|
|
|
return dataV.slice();
|
2017-06-16 09:23:50 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
public int getStrideY() {
|
2017-07-18 11:33:39 -07:00
|
|
|
return strideY;
|
2017-06-16 09:23:50 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
public int getStrideU() {
|
2017-07-18 11:33:39 -07:00
|
|
|
return strideU;
|
2017-06-16 09:23:50 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
public int getStrideV() {
|
2017-07-18 11:33:39 -07:00
|
|
|
return strideV;
|
2017-06-16 09:23:50 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
public I420Buffer toI420() {
|
2017-08-09 05:25:49 -07:00
|
|
|
retain();
|
2017-06-16 09:23:50 -07:00
|
|
|
return this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
2017-07-18 11:33:39 -07:00
|
|
|
public void retain() {
|
2017-08-21 08:02:58 -07:00
|
|
|
synchronized (refCountLock) {
|
|
|
|
|
++refCount;
|
|
|
|
|
}
|
2017-07-18 11:33:39 -07:00
|
|
|
}
|
2017-06-16 09:23:50 -07:00
|
|
|
|
|
|
|
|
@Override
|
2017-07-18 11:33:39 -07:00
|
|
|
public void release() {
|
2017-08-21 08:02:58 -07:00
|
|
|
synchronized (refCountLock) {
|
|
|
|
|
if (--refCount == 0 && releaseCallback != null) {
|
|
|
|
|
releaseCallback.run();
|
|
|
|
|
}
|
2017-07-18 11:33:39 -07:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-28 07:12:23 -07:00
|
|
|
@Override
|
|
|
|
|
public VideoFrame.Buffer cropAndScale(
|
|
|
|
|
int cropX, int cropY, int cropWidth, int cropHeight, int scaleWidth, int scaleHeight) {
|
|
|
|
|
return VideoFrame.cropAndScaleI420(
|
|
|
|
|
this, cropX, cropY, cropWidth, cropHeight, scaleWidth, scaleHeight);
|
2017-07-18 11:33:39 -07:00
|
|
|
}
|
2017-06-16 09:23:50 -07:00
|
|
|
}
|