package org.jcodec.codecs.h264.decode;
import static org.jcodec.codecs.h264.H264Const.identityMapping4;
import static org.jcodec.codecs.h264.H264Const.PartPred.L0;
import static org.jcodec.codecs.h264.decode.MBlockDecoderUtils.calcMVPredictionMedian;
import static org.jcodec.codecs.h264.decode.MBlockDecoderUtils.collectPredictors;
import static org.jcodec.codecs.h264.decode.MBlockDecoderUtils.copyVect;
import static org.jcodec.codecs.h264.decode.MBlockDecoderUtils.saveMvs;
import static org.jcodec.codecs.h264.decode.MBlockDecoderUtils.savePrediction8x8;
import static org.jcodec.codecs.h264.decode.MBlockDecoderUtils.saveVect;
import static org.jcodec.codecs.h264.io.model.SliceType.P;
import org.jcodec.codecs.h264.H264Const.PartPred;
import org.jcodec.codecs.h264.decode.aso.Mapper;
import org.jcodec.codecs.h264.io.model.Frame;
import org.jcodec.codecs.h264.io.model.SliceHeader;
import org.jcodec.codecs.h264.io.model.SliceType;
import org.jcodec.common.model.Picture8Bit;
import java.util.Arrays;
/**
* A decoder for P skip macroblocks
*
* @author The JCodec project
*/
public class MBlockSkipDecoder extends MBlockDecoderBase {
private Mapper mapper;
private MBlockDecoderBDirect bDirectDecoder;
public MBlockSkipDecoder(Mapper mapper, MBlockDecoderBDirect bDirectDecoder,
SliceHeader sh, DeblockerInput di, int poc, DecoderState sharedState) {
super(sh, di, poc, sharedState);
this.mapper = mapper;
this.bDirectDecoder = bDirectDecoder;
}
public void decodeSkip(MBlock mBlock, Frame[][] refs, Picture8Bit mb, SliceType sliceType) {
int mbX = mapper.getMbX(mBlock.mbIdx);
int mbY = mapper.getMbY(mBlock.mbIdx);
int mbAddr = mapper.getAddress(mBlock.mbIdx);
int[][][] x = new int[2][16][3];
PartPred[] pp = new PartPred[4];
for (int i = 0; i < 16; i++)
x[0][i][2] = x[1][i][2] = -1;
if (sliceType == P) {
predictPSkip(refs, mbX, mbY, mapper.leftAvailable(mBlock.mbIdx), mapper.topAvailable(mBlock.mbIdx),
mapper.topLeftAvailable(mBlock.mbIdx), mapper.topRightAvailable(mBlock.mbIdx), x, mb);
Arrays.fill(pp, PartPred.L0);
} else {
bDirectDecoder.predictBDirect(refs, mbX, mbY, mapper.leftAvailable(mBlock.mbIdx),
mapper.topAvailable(mBlock.mbIdx), mapper.topLeftAvailable(mBlock.mbIdx),
mapper.topRightAvailable(mBlock.mbIdx), x, pp, mb, identityMapping4);
savePrediction8x8(s, mbX, x[0], 0);
savePrediction8x8(s, mbX, x[1], 1);
}
decodeChromaSkip(refs, x, pp, mbX, mbY, mb);
collectPredictors(s, mb, mbX);
saveMvs(di, x, mbX, mbY);
di.mbTypes[mbAddr] = mBlock.curMbType;
di.mbQps[0][mbAddr] = s.qp;
di.mbQps[1][mbAddr] = calcQpChroma(s.qp, s.chromaQpOffset[0]);
di.mbQps[2][mbAddr] = calcQpChroma(s.qp, s.chromaQpOffset[1]);
}
public void predictPSkip(Frame[][] refs, int mbX, int mbY, boolean lAvb, boolean tAvb, boolean tlAvb,
boolean trAvb, int[][][] x, Picture8Bit mb) {
int mvX = 0, mvY = 0;
if (lAvb && tAvb) {
int[] b = s.mvTop[0][mbX << 2];
int[] a = s.mvLeft[0][0];
if ((a[0] != 0 || a[1] != 0 || a[2] != 0) && (b[0] != 0 || b[1] != 0 || b[2] != 0)) {
mvX = calcMVPredictionMedian(a, b, s.mvTop[0][(mbX << 2) + 4], s.mvTopLeft[0], lAvb, tAvb, trAvb,
tlAvb, 0, 0);
mvY = calcMVPredictionMedian(a, b, s.mvTop[0][(mbX << 2) + 4], s.mvTopLeft[0], lAvb, tAvb, trAvb,
tlAvb, 0, 1);
}
}
int xx = mbX << 2;
copyVect(s.mvTopLeft[0], s.mvTop[0][xx + 3]);
saveVect(s.mvTop[0], xx, xx + 4, mvX, mvY, 0);
saveVect(s.mvLeft[0], 0, 4, mvX, mvY, 0);
for (int i = 0; i < 16; i++) {
x[0][i][0] = mvX;
x[0][i][1] = mvY;
x[0][i][2] = 0;
}
interpolator.getBlockLuma(refs[0][0], mb, 0, (mbX << 6) + mvX, (mbY << 6) + mvY, 16, 16);
PredictionMerger.mergePrediction(sh, 0, 0, L0, 0, mb.getPlaneData(0), null, 0, 16, 16, 16, mb.getPlaneData(0),
refs, poc);
}
public void decodeChromaSkip(Frame[][] reference, int[][][] vectors, PartPred[] pp, int mbX, int mbY, Picture8Bit mb) {
predictChromaInter(reference, vectors, mbX << 3, mbY << 3, 1, mb, pp);
predictChromaInter(reference, vectors, mbX << 3, mbY << 3, 2, mb, pp);
}
}