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 android.graphics; 18 19 import android.compat.annotation.UnsupportedAppUsage; 20 import android.hardware.HardwareBuffer; 21 import android.os.Build; 22 import android.os.Parcel; 23 import android.os.Parcelable; 24 25 /** 26 * Simple wrapper for the native GraphicBuffer class. 27 * 28 * @hide 29 */ 30 @SuppressWarnings("UnusedDeclaration") 31 public class GraphicBuffer implements Parcelable { 32 // Note: keep usage flags in sync with GraphicBuffer.h and gralloc.h 33 public static final int USAGE_SW_READ_NEVER = 0x0; 34 public static final int USAGE_SW_READ_RARELY = 0x2; 35 public static final int USAGE_SW_READ_OFTEN = 0x3; 36 public static final int USAGE_SW_READ_MASK = 0xF; 37 38 public static final int USAGE_SW_WRITE_NEVER = 0x0; 39 public static final int USAGE_SW_WRITE_RARELY = 0x20; 40 public static final int USAGE_SW_WRITE_OFTEN = 0x30; 41 public static final int USAGE_SW_WRITE_MASK = 0xF0; 42 43 public static final int USAGE_SOFTWARE_MASK = USAGE_SW_READ_MASK | USAGE_SW_WRITE_MASK; 44 45 public static final int USAGE_PROTECTED = 0x4000; 46 47 public static final int USAGE_HW_TEXTURE = 0x100; 48 public static final int USAGE_HW_RENDER = 0x200; 49 public static final int USAGE_HW_2D = 0x400; 50 public static final int USAGE_HW_COMPOSER = 0x800; 51 public static final int USAGE_HW_VIDEO_ENCODER = 0x10000; 52 public static final int USAGE_HW_MASK = 0x71F00; 53 54 private final int mWidth; 55 private final int mHeight; 56 private final int mFormat; 57 private final int mUsage; 58 // Note: do not rename, this field is used by native code 59 @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553) 60 private long mNativeObject; 61 62 // These two fields are only used by lock/unlockCanvas() 63 private Canvas mCanvas; 64 private int mSaveCount; 65 66 // If set to true, this GraphicBuffer instance cannot be used anymore 67 private boolean mDestroyed; 68 69 /** 70 * Creates new <code>GraphicBuffer</code> instance. This method will return null 71 * if the buffer cannot be created. 72 * 73 * @param width The width in pixels of the buffer 74 * @param height The height in pixels of the buffer 75 * @param format The format of each pixel as specified in {@link PixelFormat} 76 * @param usage Hint indicating how the buffer will be used 77 * 78 * @return A <code>GraphicBuffer</code> instance or null 79 */ create(int width, int height, int format, int usage)80 public static GraphicBuffer create(int width, int height, int format, int usage) { 81 long nativeObject = nCreateGraphicBuffer(width, height, format, usage); 82 if (nativeObject != 0) { 83 return new GraphicBuffer(width, height, format, usage, nativeObject); 84 } 85 return null; 86 } 87 88 /** 89 * Private use only. See {@link #create(int, int, int, int)}. 90 */ 91 @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553) GraphicBuffer(int width, int height, int format, int usage, long nativeObject)92 private GraphicBuffer(int width, int height, int format, int usage, long nativeObject) { 93 mWidth = width; 94 mHeight = height; 95 mFormat = format; 96 mUsage = usage; 97 mNativeObject = nativeObject; 98 } 99 100 /** 101 * For Bitmap until all usages are updated to AHB 102 * @hide 103 */ createFromHardwareBuffer(HardwareBuffer buffer)104 public static final GraphicBuffer createFromHardwareBuffer(HardwareBuffer buffer) { 105 return nCreateFromHardwareBuffer(buffer); 106 } 107 108 /** 109 * Returns the width of this buffer in pixels. 110 */ getWidth()111 public int getWidth() { 112 return mWidth; 113 } 114 115 /** 116 * Returns the height of this buffer in pixels. 117 */ getHeight()118 public int getHeight() { 119 return mHeight; 120 } 121 122 /** 123 * Returns the pixel format of this buffer. The pixel format must be one of 124 * the formats defined in {@link PixelFormat}. 125 */ getFormat()126 public int getFormat() { 127 return mFormat; 128 } 129 130 /** 131 * Returns the usage hint set on this buffer. 132 */ getUsage()133 public int getUsage() { 134 return mUsage; 135 } 136 137 /** 138 * <p>Start editing the pixels in the buffer. A null is returned if the buffer 139 * cannot be locked for editing.</p> 140 * 141 * <p>The content of the buffer is preserved between unlockCanvas() 142 * and lockCanvas().</p> 143 * 144 * <p>If this method is called after {@link #destroy()}, the return value will 145 * always be null.</p> 146 * 147 * @return A Canvas used to draw into the buffer, or null. 148 * 149 * @see #lockCanvas(android.graphics.Rect) 150 * @see #unlockCanvasAndPost(android.graphics.Canvas) 151 * @see #isDestroyed() 152 */ lockCanvas()153 public Canvas lockCanvas() { 154 return lockCanvas(null); 155 } 156 157 /** 158 * Just like {@link #lockCanvas()} but allows specification of a dirty 159 * rectangle. 160 * 161 * <p>If this method is called after {@link #destroy()}, the return value will 162 * always be null.</p> 163 * 164 * @param dirty Area of the buffer that may be modified. 165 166 * @return A Canvas used to draw into the surface, or null. 167 * 168 * @see #lockCanvas() 169 * @see #unlockCanvasAndPost(android.graphics.Canvas) 170 * @see #isDestroyed() 171 */ lockCanvas(Rect dirty)172 public Canvas lockCanvas(Rect dirty) { 173 if (mDestroyed) { 174 return null; 175 } 176 177 if (mCanvas == null) { 178 mCanvas = new Canvas(); 179 } 180 181 if (nLockCanvas(mNativeObject, mCanvas, dirty)) { 182 mSaveCount = mCanvas.save(); 183 return mCanvas; 184 } 185 186 return null; 187 } 188 189 /** 190 * Finish editing pixels in the buffer. 191 * 192 * <p>This method doesn't do anything if {@link #destroy()} was 193 * previously called.</p> 194 * 195 * @param canvas The Canvas previously returned by lockCanvas() 196 * 197 * @see #lockCanvas() 198 * @see #lockCanvas(android.graphics.Rect) 199 * @see #isDestroyed() 200 */ unlockCanvasAndPost(Canvas canvas)201 public void unlockCanvasAndPost(Canvas canvas) { 202 if (!mDestroyed && mCanvas != null && canvas == mCanvas) { 203 canvas.restoreToCount(mSaveCount); 204 mSaveCount = 0; 205 206 nUnlockCanvasAndPost(mNativeObject, mCanvas); 207 } 208 } 209 210 /** 211 * Destroyes this buffer immediately. Calling this method frees up any 212 * underlying native resources. After calling this method, this buffer 213 * must not be used in any way ({@link #lockCanvas()} must not be called, 214 * etc.) 215 * 216 * @see #isDestroyed() 217 */ destroy()218 public void destroy() { 219 if (!mDestroyed) { 220 mDestroyed = true; 221 nDestroyGraphicBuffer(mNativeObject); 222 mNativeObject = 0; 223 } 224 } 225 226 /** 227 * Indicates whether this buffer has been destroyed. A destroyed buffer 228 * cannot be used in any way: locking a Canvas will return null, the buffer 229 * cannot be written to a parcel, etc. 230 * 231 * @return True if this <code>GraphicBuffer</code> is in a destroyed state, 232 * false otherwise. 233 * 234 * @see #destroy() 235 */ isDestroyed()236 public boolean isDestroyed() { 237 return mDestroyed; 238 } 239 240 @Override finalize()241 protected void finalize() throws Throwable { 242 try { 243 destroy(); 244 } finally { 245 super.finalize(); 246 } 247 } 248 249 @Override describeContents()250 public int describeContents() { 251 return 0; 252 } 253 254 /** 255 * Flatten this object in to a Parcel. 256 * 257 * <p>Calling this method will throw an <code>IllegalStateException</code> if 258 * {@link #destroy()} has been previously called.</p> 259 * 260 * @param dest The Parcel in which the object should be written. 261 * @param flags Additional flags about how the object should be written. 262 * May be 0 or {@link #PARCELABLE_WRITE_RETURN_VALUE}. 263 */ 264 @Override writeToParcel(Parcel dest, int flags)265 public void writeToParcel(Parcel dest, int flags) { 266 if (mDestroyed) { 267 throw new IllegalStateException("This GraphicBuffer has been destroyed and cannot be " 268 + "written to a parcel."); 269 } 270 271 dest.writeInt(mWidth); 272 dest.writeInt(mHeight); 273 dest.writeInt(mFormat); 274 dest.writeInt(mUsage); 275 nWriteGraphicBufferToParcel(mNativeObject, dest); 276 } 277 278 @UnsupportedAppUsage 279 public static final @android.annotation.NonNull Parcelable.Creator<GraphicBuffer> CREATOR = 280 new Parcelable.Creator<GraphicBuffer>() { 281 public GraphicBuffer createFromParcel(Parcel in) { 282 int width = in.readInt(); 283 int height = in.readInt(); 284 int format = in.readInt(); 285 int usage = in.readInt(); 286 long nativeObject = nReadGraphicBufferFromParcel(in); 287 if (nativeObject != 0) { 288 return new GraphicBuffer(width, height, format, usage, nativeObject); 289 } 290 return null; 291 } 292 293 public GraphicBuffer[] newArray(int size) { 294 return new GraphicBuffer[size]; 295 } 296 }; 297 nCreateGraphicBuffer(int width, int height, int format, int usage)298 private static native long nCreateGraphicBuffer(int width, int height, int format, int usage); nDestroyGraphicBuffer(long nativeObject)299 private static native void nDestroyGraphicBuffer(long nativeObject); nWriteGraphicBufferToParcel(long nativeObject, Parcel dest)300 private static native void nWriteGraphicBufferToParcel(long nativeObject, Parcel dest); nReadGraphicBufferFromParcel(Parcel in)301 private static native long nReadGraphicBufferFromParcel(Parcel in); nLockCanvas(long nativeObject, Canvas canvas, Rect dirty)302 private static native boolean nLockCanvas(long nativeObject, Canvas canvas, Rect dirty); nUnlockCanvasAndPost(long nativeObject, Canvas canvas)303 private static native boolean nUnlockCanvasAndPost(long nativeObject, Canvas canvas); nCreateFromHardwareBuffer(HardwareBuffer buffer)304 private static native GraphicBuffer nCreateFromHardwareBuffer(HardwareBuffer buffer); 305 } 306