Home | History | Annotate | Download | only in pipeline
      1 /*
      2  * Copyright (C) 2013 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.pipeline;
     18 
     19 import android.content.Context;
     20 import android.content.res.Resources;
     21 import android.graphics.Bitmap;
     22 import android.graphics.Canvas;
     23 import android.graphics.Matrix;
     24 import android.graphics.Paint;
     25 import android.graphics.Rect;
     26 import android.graphics.RectF;
     27 import android.support.v8.renderscript.Allocation;
     28 import android.support.v8.renderscript.RenderScript;
     29 import android.util.Log;
     30 
     31 import com.android.gallery3d.filtershow.cache.BitmapCache;
     32 import com.android.gallery3d.filtershow.cache.ImageLoader;
     33 import com.android.gallery3d.filtershow.filters.FilterRepresentation;
     34 import com.android.gallery3d.filtershow.filters.FiltersManager;
     35 import com.android.gallery3d.filtershow.imageshow.GeometryMathUtils;
     36 import com.android.gallery3d.filtershow.imageshow.MasterImage;
     37 
     38 import java.util.Vector;
     39 
     40 public class CachingPipeline implements PipelineInterface {
     41     private static final String LOGTAG = "CachingPipeline";
     42     private boolean DEBUG = false;
     43 
     44     private static final Bitmap.Config BITMAP_CONFIG = Bitmap.Config.ARGB_8888;
     45 
     46     private static volatile RenderScript sRS = null;
     47 
     48     private FiltersManager mFiltersManager = null;
     49     private volatile Bitmap mOriginalBitmap = null;
     50     private volatile Bitmap mResizedOriginalBitmap = null;
     51 
     52     private FilterEnvironment mEnvironment = new FilterEnvironment();
     53     private CacheProcessing mCachedProcessing = new CacheProcessing();
     54 
     55 
     56     private volatile Allocation mOriginalAllocation = null;
     57     private volatile Allocation mFiltersOnlyOriginalAllocation =  null;
     58 
     59     protected volatile Allocation mInPixelsAllocation;
     60     protected volatile Allocation mOutPixelsAllocation;
     61     private volatile int mWidth = 0;
     62     private volatile int mHeight = 0;
     63 
     64     private volatile float mPreviewScaleFactor = 1.0f;
     65     private volatile float mHighResPreviewScaleFactor = 1.0f;
     66     private volatile String mName = "";
     67 
     68     public CachingPipeline(FiltersManager filtersManager, String name) {
     69         mFiltersManager = filtersManager;
     70         mName = name;
     71     }
     72 
     73     public static synchronized RenderScript getRenderScriptContext() {
     74         return sRS;
     75     }
     76 
     77     public static synchronized void createRenderscriptContext(Context context) {
     78         if (sRS != null) {
     79             Log.w(LOGTAG, "A prior RS context exists when calling setRenderScriptContext");
     80             destroyRenderScriptContext();
     81         }
     82         sRS = RenderScript.create(context);
     83     }
     84 
     85     public static synchronized void destroyRenderScriptContext() {
     86         if (sRS != null) {
     87             sRS.destroy();
     88         }
     89         sRS = null;
     90     }
     91 
     92     public void stop() {
     93         mEnvironment.setStop(true);
     94     }
     95 
     96     public synchronized void reset() {
     97         synchronized (CachingPipeline.class) {
     98             if (getRenderScriptContext() == null) {
     99                 return;
    100             }
    101             mOriginalBitmap = null; // just a reference to the bitmap in ImageLoader
    102             if (mResizedOriginalBitmap != null) {
    103                 mResizedOriginalBitmap.recycle();
    104                 mResizedOriginalBitmap = null;
    105             }
    106             if (mOriginalAllocation != null) {
    107                 mOriginalAllocation.destroy();
    108                 mOriginalAllocation = null;
    109             }
    110             if (mFiltersOnlyOriginalAllocation != null) {
    111                 mFiltersOnlyOriginalAllocation.destroy();
    112                 mFiltersOnlyOriginalAllocation = null;
    113             }
    114             mPreviewScaleFactor = 1.0f;
    115             mHighResPreviewScaleFactor = 1.0f;
    116 
    117             destroyPixelAllocations();
    118         }
    119     }
    120 
    121     public Resources getResources() {
    122         return sRS.getApplicationContext().getResources();
    123     }
    124 
    125     private synchronized void destroyPixelAllocations() {
    126         if (DEBUG) {
    127             Log.v(LOGTAG, "destroyPixelAllocations in " + getName());
    128         }
    129         if (mInPixelsAllocation != null) {
    130             mInPixelsAllocation.destroy();
    131             mInPixelsAllocation = null;
    132         }
    133         if (mOutPixelsAllocation != null) {
    134             mOutPixelsAllocation.destroy();
    135             mOutPixelsAllocation = null;
    136         }
    137         mWidth = 0;
    138         mHeight = 0;
    139     }
    140 
    141     private String getType(RenderingRequest request) {
    142         if (request.getType() == RenderingRequest.ICON_RENDERING) {
    143             return "ICON_RENDERING";
    144         }
    145         if (request.getType() == RenderingRequest.FILTERS_RENDERING) {
    146             return "FILTERS_RENDERING";
    147         }
    148         if (request.getType() == RenderingRequest.FULL_RENDERING) {
    149             return "FULL_RENDERING";
    150         }
    151         if (request.getType() == RenderingRequest.GEOMETRY_RENDERING) {
    152             return "GEOMETRY_RENDERING";
    153         }
    154         if (request.getType() == RenderingRequest.PARTIAL_RENDERING) {
    155             return "PARTIAL_RENDERING";
    156         }
    157         if (request.getType() == RenderingRequest.HIGHRES_RENDERING) {
    158             return "HIGHRES_RENDERING";
    159         }
    160         return "UNKNOWN TYPE!";
    161     }
    162 
    163     private void setupEnvironment(ImagePreset preset, boolean highResPreview) {
    164         mEnvironment.setPipeline(this);
    165         mEnvironment.setFiltersManager(mFiltersManager);
    166         mEnvironment.setBitmapCache(MasterImage.getImage().getBitmapCache());
    167         if (highResPreview) {
    168             mEnvironment.setScaleFactor(mHighResPreviewScaleFactor);
    169         } else {
    170             mEnvironment.setScaleFactor(mPreviewScaleFactor);
    171         }
    172         mEnvironment.setQuality(FilterEnvironment.QUALITY_PREVIEW);
    173         mEnvironment.setImagePreset(preset);
    174         mEnvironment.setStop(false);
    175     }
    176 
    177     public void setOriginal(Bitmap bitmap) {
    178         mOriginalBitmap = bitmap;
    179         Log.v(LOGTAG,"setOriginal, size " + bitmap.getWidth() + " x " + bitmap.getHeight());
    180         ImagePreset preset = MasterImage.getImage().getPreset();
    181         setupEnvironment(preset, false);
    182         updateOriginalAllocation(preset);
    183     }
    184 
    185     private synchronized boolean updateOriginalAllocation(ImagePreset preset) {
    186         if (preset == null) {
    187             return false;
    188         }
    189         Bitmap originalBitmap = mOriginalBitmap;
    190 
    191         if (originalBitmap == null) {
    192             return false;
    193         }
    194 
    195         RenderScript RS = getRenderScriptContext();
    196 
    197         Allocation filtersOnlyOriginalAllocation = mFiltersOnlyOriginalAllocation;
    198         mFiltersOnlyOriginalAllocation = Allocation.createFromBitmap(RS, originalBitmap,
    199                 Allocation.MipmapControl.MIPMAP_NONE, Allocation.USAGE_SCRIPT);
    200         if (filtersOnlyOriginalAllocation != null) {
    201             filtersOnlyOriginalAllocation.destroy();
    202         }
    203 
    204         Allocation originalAllocation = mOriginalAllocation;
    205         mResizedOriginalBitmap = preset.applyGeometry(originalBitmap, mEnvironment);
    206         mOriginalAllocation = Allocation.createFromBitmap(RS, mResizedOriginalBitmap,
    207                 Allocation.MipmapControl.MIPMAP_NONE, Allocation.USAGE_SCRIPT);
    208         if (originalAllocation != null) {
    209             originalAllocation.destroy();
    210         }
    211 
    212         return true;
    213     }
    214 
    215     public void renderHighres(RenderingRequest request) {
    216         synchronized (CachingPipeline.class) {
    217             if (getRenderScriptContext() == null) {
    218                 return;
    219             }
    220             ImagePreset preset = request.getImagePreset();
    221             setupEnvironment(preset, false);
    222             Bitmap bitmap = MasterImage.getImage().getOriginalBitmapHighres();
    223             if (bitmap == null) {
    224                 return;
    225             }
    226             bitmap = mEnvironment.getBitmapCopy(bitmap, BitmapCache.HIGHRES);
    227             bitmap = preset.applyGeometry(bitmap, mEnvironment);
    228 
    229             mEnvironment.setQuality(FilterEnvironment.QUALITY_PREVIEW);
    230             Bitmap bmp = preset.apply(bitmap, mEnvironment);
    231             if (!mEnvironment.needsStop()) {
    232                 request.setBitmap(bmp);
    233             } else {
    234                 mEnvironment.cache(bmp);
    235             }
    236             mFiltersManager.freeFilterResources(preset);
    237         }
    238     }
    239 
    240     public void renderGeometry(RenderingRequest request) {
    241         synchronized (CachingPipeline.class) {
    242             if (getRenderScriptContext() == null) {
    243                 return;
    244             }
    245             ImagePreset preset = request.getImagePreset();
    246             setupEnvironment(preset, false);
    247             Bitmap bitmap = MasterImage.getImage().getOriginalBitmapHighres();
    248             if (bitmap == null) {
    249                 return;
    250             }
    251             bitmap = mEnvironment.getBitmapCopy(bitmap, BitmapCache.GEOMETRY);
    252             bitmap = preset.applyGeometry(bitmap, mEnvironment);
    253             if (!mEnvironment.needsStop()) {
    254                 request.setBitmap(bitmap);
    255             } else {
    256                 mEnvironment.cache(bitmap);
    257             }
    258             mFiltersManager.freeFilterResources(preset);
    259         }
    260     }
    261 
    262     public void renderFilters(RenderingRequest request) {
    263         synchronized (CachingPipeline.class) {
    264             if (getRenderScriptContext() == null) {
    265                 return;
    266             }
    267             ImagePreset preset = request.getImagePreset();
    268             setupEnvironment(preset, false);
    269             Bitmap bitmap = MasterImage.getImage().getOriginalBitmapHighres();
    270             if (bitmap == null) {
    271                 return;
    272             }
    273             bitmap = mEnvironment.getBitmapCopy(bitmap, BitmapCache.FILTERS);
    274             bitmap = preset.apply(bitmap, mEnvironment);
    275             if (!mEnvironment.needsStop()) {
    276                 request.setBitmap(bitmap);
    277             } else {
    278                 mEnvironment.cache(bitmap);
    279             }
    280             mFiltersManager.freeFilterResources(preset);
    281         }
    282     }
    283 
    284     public synchronized void render(RenderingRequest request) {
    285         // TODO: cleanup/remove GEOMETRY / FILTERS paths
    286         synchronized (CachingPipeline.class) {
    287             if (getRenderScriptContext() == null) {
    288                 return;
    289             }
    290             if ((request.getType() != RenderingRequest.PARTIAL_RENDERING
    291                   && request.getType() != RenderingRequest.ICON_RENDERING
    292                     && request.getBitmap() == null)
    293                     || request.getImagePreset() == null) {
    294                 return;
    295             }
    296 
    297             if (DEBUG) {
    298                 Log.v(LOGTAG, "render image of type " + getType(request));
    299             }
    300 
    301             Bitmap bitmap = request.getBitmap();
    302             ImagePreset preset = request.getImagePreset();
    303             setupEnvironment(preset, true);
    304             mFiltersManager.freeFilterResources(preset);
    305 
    306             if (request.getType() == RenderingRequest.PARTIAL_RENDERING) {
    307                 MasterImage master = MasterImage.getImage();
    308                 bitmap = ImageLoader.getScaleOneImageForPreset(master.getActivity(),
    309                         mEnvironment.getBimapCache(),
    310                         master.getUri(), request.getBounds(),
    311                         request.getDestination());
    312                 if (bitmap == null) {
    313                     Log.w(LOGTAG, "could not get bitmap for: " + getType(request));
    314                     return;
    315                 }
    316             }
    317 
    318             if (request.getType() == RenderingRequest.FULL_RENDERING
    319                     || request.getType() == RenderingRequest.GEOMETRY_RENDERING
    320                     || request.getType() == RenderingRequest.FILTERS_RENDERING) {
    321                 updateOriginalAllocation(preset);
    322             }
    323 
    324             if (DEBUG && bitmap != null) {
    325                 Log.v(LOGTAG, "after update, req bitmap (" + bitmap.getWidth() + "x" + bitmap.getHeight()
    326                         + " ? resizeOriginal (" + mResizedOriginalBitmap.getWidth() + "x"
    327                         + mResizedOriginalBitmap.getHeight());
    328             }
    329 
    330             if (request.getType() == RenderingRequest.FULL_RENDERING
    331                     || request.getType() == RenderingRequest.GEOMETRY_RENDERING) {
    332                 mOriginalAllocation.copyTo(bitmap);
    333             } else if (request.getType() == RenderingRequest.FILTERS_RENDERING) {
    334                 mFiltersOnlyOriginalAllocation.copyTo(bitmap);
    335             }
    336 
    337             if (request.getType() == RenderingRequest.FULL_RENDERING
    338                     || request.getType() == RenderingRequest.FILTERS_RENDERING
    339                     || request.getType() == RenderingRequest.ICON_RENDERING
    340                     || request.getType() == RenderingRequest.PARTIAL_RENDERING
    341                     || request.getType() == RenderingRequest.STYLE_ICON_RENDERING) {
    342 
    343                 if (request.getType() == RenderingRequest.ICON_RENDERING) {
    344                     mEnvironment.setQuality(FilterEnvironment.QUALITY_ICON);
    345                 } else {
    346                     mEnvironment.setQuality(FilterEnvironment.QUALITY_PREVIEW);
    347                 }
    348 
    349                 if (request.getType() == RenderingRequest.ICON_RENDERING) {
    350                     Rect iconBounds = request.getIconBounds();
    351                     Bitmap source = MasterImage.getImage().getThumbnailBitmap();
    352                     if (iconBounds.width() > source.getWidth() * 2) {
    353                         source = MasterImage.getImage().getLargeThumbnailBitmap();
    354                     }
    355                     if (iconBounds != null) {
    356                         bitmap = mEnvironment.getBitmap(iconBounds.width(),
    357                                 iconBounds.height(), BitmapCache.ICON);
    358                         Canvas canvas = new Canvas(bitmap);
    359                         Matrix m = new Matrix();
    360                         float minSize = Math.min(source.getWidth(), source.getHeight());
    361                         float maxSize = Math.max(iconBounds.width(), iconBounds.height());
    362                         float scale = maxSize / minSize;
    363                         m.setScale(scale, scale);
    364                         float dx = (iconBounds.width() - (source.getWidth() * scale))/2.0f;
    365                         float dy = (iconBounds.height() - (source.getHeight() * scale))/2.0f;
    366                         m.postTranslate(dx, dy);
    367                         canvas.drawBitmap(source, m, new Paint(Paint.FILTER_BITMAP_FLAG));
    368                     } else {
    369                         bitmap = mEnvironment.getBitmapCopy(source, BitmapCache.ICON);
    370                     }
    371                 }
    372                 Bitmap bmp = preset.apply(bitmap, mEnvironment);
    373                 if (!mEnvironment.needsStop()) {
    374                     request.setBitmap(bmp);
    375                 }
    376                 mFiltersManager.freeFilterResources(preset);
    377             }
    378         }
    379     }
    380 
    381     public synchronized void renderImage(ImagePreset preset, Allocation in, Allocation out) {
    382         synchronized (CachingPipeline.class) {
    383             if (getRenderScriptContext() == null) {
    384                 return;
    385             }
    386             setupEnvironment(preset, false);
    387             mFiltersManager.freeFilterResources(preset);
    388             preset.applyFilters(-1, -1, in, out, mEnvironment);
    389             boolean copyOut = false;
    390             if (preset.nbFilters() > 0) {
    391                 copyOut = true;
    392             }
    393             preset.applyBorder(in, out, copyOut, mEnvironment);
    394         }
    395     }
    396 
    397     public synchronized Bitmap renderFinalImage(Bitmap bitmap, ImagePreset preset) {
    398         synchronized (CachingPipeline.class) {
    399             if (getRenderScriptContext() == null) {
    400                 return bitmap;
    401             }
    402             setupEnvironment(preset, false);
    403             mEnvironment.setQuality(FilterEnvironment.QUALITY_FINAL);
    404             mEnvironment.setScaleFactor(1.0f);
    405             mFiltersManager.freeFilterResources(preset);
    406             bitmap = preset.applyGeometry(bitmap, mEnvironment);
    407             bitmap = preset.apply(bitmap, mEnvironment);
    408             return bitmap;
    409         }
    410     }
    411 
    412     public Bitmap renderGeometryIcon(Bitmap bitmap, ImagePreset preset) {
    413         return GeometryMathUtils.applyGeometryRepresentations(preset.getGeometryFilters(), bitmap);
    414     }
    415 
    416     public void compute(SharedBuffer buffer, ImagePreset preset, int type) {
    417         if (getRenderScriptContext() == null) {
    418             return;
    419         }
    420         setupEnvironment(preset, false);
    421         Vector<FilterRepresentation> filters = preset.getFilters();
    422         Bitmap result = mCachedProcessing.process(mOriginalBitmap, filters, mEnvironment);
    423         buffer.setProducer(result);
    424         mEnvironment.cache(result);
    425     }
    426 
    427     public boolean needsRepaint() {
    428         SharedBuffer buffer = MasterImage.getImage().getPreviewBuffer();
    429         return buffer.checkRepaintNeeded();
    430     }
    431 
    432     public void setPreviewScaleFactor(float previewScaleFactor) {
    433         mPreviewScaleFactor = previewScaleFactor;
    434     }
    435 
    436     public void setHighResPreviewScaleFactor(float highResPreviewScaleFactor) {
    437         mHighResPreviewScaleFactor = highResPreviewScaleFactor;
    438     }
    439 
    440     public synchronized boolean isInitialized() {
    441         return getRenderScriptContext() != null && mOriginalBitmap != null;
    442     }
    443 
    444     public boolean prepareRenderscriptAllocations(Bitmap bitmap) {
    445         RenderScript RS = getRenderScriptContext();
    446         boolean needsUpdate = false;
    447         if (mOutPixelsAllocation == null || mInPixelsAllocation == null ||
    448                 bitmap.getWidth() != mWidth || bitmap.getHeight() != mHeight) {
    449             destroyPixelAllocations();
    450             Bitmap bitmapBuffer = bitmap;
    451             if (bitmap.getConfig() == null || bitmap.getConfig() != BITMAP_CONFIG) {
    452                 bitmapBuffer = bitmap.copy(BITMAP_CONFIG, true);
    453             }
    454             mOutPixelsAllocation = Allocation.createFromBitmap(RS, bitmapBuffer,
    455                     Allocation.MipmapControl.MIPMAP_NONE, Allocation.USAGE_SCRIPT);
    456             mInPixelsAllocation = Allocation.createTyped(RS,
    457                     mOutPixelsAllocation.getType());
    458             needsUpdate = true;
    459         }
    460         if (RS != null) {
    461             mInPixelsAllocation.copyFrom(bitmap);
    462         }
    463         if (bitmap.getWidth() != mWidth
    464                 || bitmap.getHeight() != mHeight) {
    465             mWidth = bitmap.getWidth();
    466             mHeight = bitmap.getHeight();
    467             needsUpdate = true;
    468         }
    469         if (DEBUG) {
    470             Log.v(LOGTAG, "prepareRenderscriptAllocations: " + needsUpdate + " in " + getName());
    471         }
    472         return needsUpdate;
    473     }
    474 
    475     public synchronized Allocation getInPixelsAllocation() {
    476         return mInPixelsAllocation;
    477     }
    478 
    479     public synchronized Allocation getOutPixelsAllocation() {
    480         return mOutPixelsAllocation;
    481     }
    482 
    483     public String getName() {
    484         return mName;
    485     }
    486 
    487     public RenderScript getRSContext() {
    488         return CachingPipeline.getRenderScriptContext();
    489     }
    490 }
    491