1 /* 2 * Copyright (C) 2012 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package com.android.gallery3d.filtershow.filters; 18 19 import android.graphics.Bitmap; 20 import android.graphics.BitmapFactory; 21 import android.renderscript.*; 22 import android.util.Log; 23 import android.content.res.Resources; 24 import com.android.gallery3d.R; 25 import com.android.gallery3d.filtershow.pipeline.PipelineInterface; 26 27 public abstract class ImageFilterRS extends ImageFilter { 28 private static final String LOGTAG = "ImageFilterRS"; 29 private boolean DEBUG = false; 30 private int mLastInputWidth = 0; 31 private int mLastInputHeight = 0; 32 private long mLastTimeCalled; 33 34 public static boolean PERF_LOGGING = false; 35 36 private static ScriptC_grey mGreyConvert = null; 37 private static RenderScript mRScache = null; 38 39 private volatile boolean mResourcesLoaded = false; 40 createFilter(android.content.res.Resources res, float scaleFactor, int quality)41 protected abstract void createFilter(android.content.res.Resources res, 42 float scaleFactor, int quality); 43 createFilter(android.content.res.Resources res, float scaleFactor, int quality, Allocation in)44 protected void createFilter(android.content.res.Resources res, 45 float scaleFactor, int quality, Allocation in) {} bindScriptValues(Allocation in)46 protected void bindScriptValues(Allocation in) {} 47 runFilter()48 protected abstract void runFilter(); 49 update(Bitmap bitmap)50 protected void update(Bitmap bitmap) { 51 getOutPixelsAllocation().copyTo(bitmap); 52 } 53 getRenderScriptContext()54 protected RenderScript getRenderScriptContext() { 55 PipelineInterface pipeline = getEnvironment().getPipeline(); 56 return pipeline.getRSContext(); 57 } 58 getInPixelsAllocation()59 protected Allocation getInPixelsAllocation() { 60 PipelineInterface pipeline = getEnvironment().getPipeline(); 61 return pipeline.getInPixelsAllocation(); 62 } 63 getOutPixelsAllocation()64 protected Allocation getOutPixelsAllocation() { 65 PipelineInterface pipeline = getEnvironment().getPipeline(); 66 return pipeline.getOutPixelsAllocation(); 67 } 68 69 @Override apply(Allocation in, Allocation out)70 public void apply(Allocation in, Allocation out) { 71 long startOverAll = System.nanoTime(); 72 if (PERF_LOGGING) { 73 long delay = (startOverAll - mLastTimeCalled) / 1000; 74 String msg = String.format("%s; image size %dx%d; ", getName(), 75 in.getType().getX(), in.getType().getY()); 76 msg += String.format("called after %.2f ms (%.2f FPS); ", 77 delay / 1000.f, 1000000.f / delay); 78 Log.i(LOGTAG, msg); 79 } 80 mLastTimeCalled = startOverAll; 81 long startFilter = 0; 82 long endFilter = 0; 83 if (!mResourcesLoaded) { 84 PipelineInterface pipeline = getEnvironment().getPipeline(); 85 createFilter(pipeline.getResources(), getEnvironment().getScaleFactor(), 86 getEnvironment().getQuality(), in); 87 mResourcesLoaded = true; 88 } 89 startFilter = System.nanoTime(); 90 bindScriptValues(in); 91 run(in, out); 92 if (PERF_LOGGING) { 93 getRenderScriptContext().finish(); 94 endFilter = System.nanoTime(); 95 long endOverAll = System.nanoTime(); 96 String msg = String.format("%s; image size %dx%d; ", getName(), 97 in.getType().getX(), in.getType().getY()); 98 long timeOverAll = (endOverAll - startOverAll) / 1000; 99 long timeFilter = (endFilter - startFilter) / 1000; 100 msg += String.format("over all %.2f ms (%.2f FPS); ", 101 timeOverAll / 1000.f, 1000000.f / timeOverAll); 102 msg += String.format("run filter %.2f ms (%.2f FPS)", 103 timeFilter / 1000.f, 1000000.f / timeFilter); 104 Log.i(LOGTAG, msg); 105 } 106 } 107 run(Allocation in, Allocation out)108 protected void run(Allocation in, Allocation out) {} 109 110 @Override apply(Bitmap bitmap, float scaleFactor, int quality)111 public Bitmap apply(Bitmap bitmap, float scaleFactor, int quality) { 112 if (bitmap == null || bitmap.getWidth() == 0 || bitmap.getHeight() == 0) { 113 return bitmap; 114 } 115 try { 116 PipelineInterface pipeline = getEnvironment().getPipeline(); 117 if (DEBUG) { 118 Log.v(LOGTAG, "apply filter " + getName() + " in pipeline " + pipeline.getName()); 119 } 120 Resources rsc = pipeline.getResources(); 121 boolean sizeChanged = false; 122 if (getInPixelsAllocation() != null 123 && ((getInPixelsAllocation().getType().getX() != mLastInputWidth) 124 || (getInPixelsAllocation().getType().getY() != mLastInputHeight))) { 125 sizeChanged = true; 126 } 127 if (pipeline.prepareRenderscriptAllocations(bitmap) 128 || !isResourcesLoaded() || sizeChanged) { 129 freeResources(); 130 createFilter(rsc, scaleFactor, quality); 131 setResourcesLoaded(true); 132 mLastInputWidth = getInPixelsAllocation().getType().getX(); 133 mLastInputHeight = getInPixelsAllocation().getType().getY(); 134 } 135 bindScriptValues(); 136 runFilter(); 137 update(bitmap); 138 if (DEBUG) { 139 Log.v(LOGTAG, "DONE apply filter " + getName() + " in pipeline " + pipeline.getName()); 140 } 141 } catch (android.renderscript.RSIllegalArgumentException e) { 142 Log.e(LOGTAG, "Illegal argument? " + e); 143 } catch (android.renderscript.RSRuntimeException e) { 144 Log.e(LOGTAG, "RS runtime exception ? " + e); 145 } catch (java.lang.OutOfMemoryError e) { 146 // Many of the renderscript filters allocated large (>16Mb resources) in order to apply. 147 System.gc(); 148 displayLowMemoryToast(); 149 Log.e(LOGTAG, "not enough memory for filter " + getName(), e); 150 } 151 return bitmap; 152 } 153 convertBitmap(RenderScript RS, Bitmap bitmap)154 protected static Allocation convertBitmap(RenderScript RS, Bitmap bitmap) { 155 return Allocation.createFromBitmap(RS, bitmap, 156 Allocation.MipmapControl.MIPMAP_NONE, 157 Allocation.USAGE_SCRIPT | Allocation.USAGE_GRAPHICS_TEXTURE); 158 } 159 convertRGBAtoA(RenderScript RS, Bitmap bitmap)160 private static Allocation convertRGBAtoA(RenderScript RS, Bitmap bitmap) { 161 if (RS != mRScache || mGreyConvert == null) { 162 mGreyConvert = new ScriptC_grey(RS); 163 mRScache = RS; 164 } 165 166 Type.Builder tb_a8 = new Type.Builder(RS, Element.A_8(RS)); 167 168 Allocation bitmapTemp = convertBitmap(RS, bitmap); 169 if (bitmapTemp.getType().getElement().isCompatible(Element.A_8(RS))) { 170 return bitmapTemp; 171 } 172 173 tb_a8.setX(bitmapTemp.getType().getX()); 174 tb_a8.setY(bitmapTemp.getType().getY()); 175 Allocation bitmapAlloc = Allocation.createTyped(RS, tb_a8.create(), 176 Allocation.MipmapControl.MIPMAP_NONE, 177 Allocation.USAGE_SCRIPT | Allocation.USAGE_GRAPHICS_TEXTURE); 178 mGreyConvert.forEach_RGBAtoA(bitmapTemp, bitmapAlloc); 179 bitmapTemp.destroy(); 180 return bitmapAlloc; 181 } 182 loadScaledResourceAlpha(int resource, int inSampleSize)183 public Allocation loadScaledResourceAlpha(int resource, int inSampleSize) { 184 Resources res = getEnvironment().getPipeline().getResources(); 185 final BitmapFactory.Options options = new BitmapFactory.Options(); 186 options.inSampleSize = inSampleSize; 187 Bitmap bitmap = BitmapFactory.decodeResource( 188 res, 189 resource, options); 190 Allocation ret = convertRGBAtoA(getRenderScriptContext(), bitmap); 191 bitmap.recycle(); 192 return ret; 193 } 194 loadScaledResourceAlpha(int resource, int w, int h, int inSampleSize)195 public Allocation loadScaledResourceAlpha(int resource, int w, int h, int inSampleSize) { 196 Resources res = getEnvironment().getPipeline().getResources(); 197 final BitmapFactory.Options options = new BitmapFactory.Options(); 198 options.inSampleSize = inSampleSize; 199 Bitmap bitmap = BitmapFactory.decodeResource( 200 res, 201 resource, options); 202 Bitmap resizeBitmap = Bitmap.createScaledBitmap(bitmap, w, h, true); 203 Allocation ret = convertRGBAtoA(getRenderScriptContext(), resizeBitmap); 204 resizeBitmap.recycle(); 205 bitmap.recycle(); 206 return ret; 207 } 208 loadResourceAlpha(int resource)209 public Allocation loadResourceAlpha(int resource) { 210 return loadScaledResourceAlpha(resource, 1); 211 } 212 loadResource(int resource)213 public Allocation loadResource(int resource) { 214 Resources res = getEnvironment().getPipeline().getResources(); 215 final BitmapFactory.Options options = new BitmapFactory.Options(); 216 options.inPreferredConfig = Bitmap.Config.ARGB_8888; 217 Bitmap bitmap = BitmapFactory.decodeResource( 218 res, 219 resource, options); 220 Allocation ret = convertBitmap(getRenderScriptContext(), bitmap); 221 bitmap.recycle(); 222 return ret; 223 } 224 isResourcesLoaded()225 private boolean isResourcesLoaded() { 226 return mResourcesLoaded; 227 } 228 setResourcesLoaded(boolean resourcesLoaded)229 private void setResourcesLoaded(boolean resourcesLoaded) { 230 mResourcesLoaded = resourcesLoaded; 231 } 232 233 /** 234 * Bitmaps and RS Allocations should be cleared here 235 */ resetAllocations()236 abstract protected void resetAllocations(); 237 238 /** 239 * RS Script objects (and all other RS objects) should be cleared here 240 */ resetScripts()241 public abstract void resetScripts(); 242 243 /** 244 * Scripts values should be bound here 245 */ bindScriptValues()246 abstract protected void bindScriptValues(); 247 freeResources()248 public void freeResources() { 249 if (!isResourcesLoaded()) { 250 return; 251 } 252 resetAllocations(); 253 mLastInputWidth = 0; 254 mLastInputHeight = 0; 255 setResourcesLoaded(false); 256 } 257 } 258