blob: 8bf80b2f9bc677fec83d7d641cccb94ddd390d41 [file] [log] [blame]
/*
* Copyright (C) 2011 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package android.filterpacks.imageproc;
import android.filterfw.core.Filter;
import android.filterfw.core.FilterContext;
import android.filterfw.core.Frame;
import android.filterfw.core.FrameFormat;
import android.filterfw.core.GenerateFieldPort;
import android.filterfw.core.MutableFrameFormat;
import android.filterfw.core.Program;
import android.filterfw.core.ShaderProgram;
import android.filterfw.format.ImageFormat;
public class ImageSlicer extends Filter {
@GenerateFieldPort(name = "xSlices")
private int mXSlices;
@GenerateFieldPort(name = "ySlices")
private int mYSlices;
@GenerateFieldPort(name = "padSize")
private int mPadSize;
// The current slice index from 0 to xSlices * ySlices
private int mSliceIndex;
private Frame mOriginalFrame;
private Program mProgram;
private int mInputWidth;
private int mInputHeight;
private int mSliceWidth;
private int mSliceHeight;
private int mOutputWidth;
private int mOutputHeight;
public ImageSlicer(String name) {
super(name);
mSliceIndex = 0;
}
@Override
public void setupPorts() {
addMaskedInputPort("image", ImageFormat.create(ImageFormat.COLORSPACE_RGBA,
FrameFormat.TARGET_GPU));
addOutputBasedOnInput("image", "image");
}
@Override
public FrameFormat getOutputFormat(String portName, FrameFormat inputFormat) {
return inputFormat;
}
private void calcOutputFormatForInput(Frame frame) {
// calculate the output size based on input size, xSlices, and ySlices
mInputWidth = frame.getFormat().getWidth();
mInputHeight = frame.getFormat().getHeight();
mSliceWidth = (mInputWidth + mXSlices - 1) / mXSlices;
mSliceHeight = (mInputHeight + mYSlices - 1)/ mYSlices;
mOutputWidth = mSliceWidth + mPadSize * 2;
mOutputHeight = mSliceHeight + mPadSize * 2;
}
@Override
public void process(FilterContext context) {
// Get input frame
if (mSliceIndex == 0) {
mOriginalFrame = pullInput("image");
calcOutputFormatForInput(mOriginalFrame);
}
FrameFormat inputFormat = mOriginalFrame.getFormat();
MutableFrameFormat outputFormat = inputFormat.mutableCopy();
outputFormat.setDimensions(mOutputWidth, mOutputHeight);
// Create output frame
Frame output = context.getFrameManager().newFrame(outputFormat);
// Create the program if not created already
if (mProgram == null) {
mProgram = ShaderProgram.createIdentity(context);
}
// Calculate the four corner of the source region
int xSliceIndex = mSliceIndex % mXSlices;
int ySliceIndex = mSliceIndex / mXSlices;
// TODO(rslin) : not sure shifting by 0.5 is needed.
float x0 = (xSliceIndex * mSliceWidth - mPadSize) / ((float) mInputWidth);
float y0 = (ySliceIndex * mSliceHeight - mPadSize) / ((float) mInputHeight);
((ShaderProgram) mProgram).setSourceRect(x0, y0,
((float) mOutputWidth) / mInputWidth,
((float) mOutputHeight) / mInputHeight);
// Process
mProgram.process(mOriginalFrame, output);
mSliceIndex++;
if (mSliceIndex == mXSlices * mYSlices) {
mSliceIndex = 0;
mOriginalFrame.release();
setWaitsOnInputPort("image", true);
} else {
// Retain the original frame so it can be used next time.
mOriginalFrame.retain();
setWaitsOnInputPort("image", false);
}
// Push output
pushOutput("image", output);
// Release pushed frame
output.release();
}
}