1
2 /*
3 * Copyright 2006 The Android Open Source Project
4 *
5 * Use of this source code is governed by a BSD-style license that can be
6 * found in the LICENSE file.
7 */
8
9
10 #ifndef SkBitmap_DEFINED
11 #define SkBitmap_DEFINED
12
13 #include "Sk64.h"
14 #include "SkColor.h"
15 #include "SkColorTable.h"
16 #include "SkImageInfo.h"
17 #include "SkPoint.h"
18 #include "SkRefCnt.h"
19
20 struct SkIRect;
21 struct SkRect;
22 class SkPaint;
23 class SkPixelRef;
24 class SkRegion;
25 class SkString;
26
27 class GrTexture;
28
29 /** \class SkBitmap
30
31 The SkBitmap class specifies a raster bitmap. A bitmap has an integer width
32 and height, and a format (config), and a pointer to the actual pixels.
33 Bitmaps can be drawn into a SkCanvas, but they are also used to specify the
34 target of a SkCanvas' drawing operations.
35 A const SkBitmap exposes getAddr(), which lets a caller write its pixels;
36 the constness is considered to apply to the bitmap's configuration, not
37 its contents.
38 */
39 class SK_API SkBitmap {
40 public:
41 class SK_API Allocator;
42
43 enum Config {
44 kNo_Config, //!< bitmap has not been configured
45 kA8_Config, //!< 8-bits per pixel, with only alpha specified (0 is transparent, 0xFF is opaque)
46 kIndex8_Config, //!< 8-bits per pixel, using SkColorTable to specify the colors
47 kRGB_565_Config, //!< 16-bits per pixel, (see SkColorPriv.h for packing)
48 kARGB_4444_Config, //!< 16-bits per pixel, (see SkColorPriv.h for packing)
49 kARGB_8888_Config, //!< 32-bits per pixel, (see SkColorPriv.h for packing)
50 };
51
52 // do not add this to the Config enum, otherwise the compiler will let us
53 // pass this as a valid parameter for Config.
54 enum {
55 kConfigCount = kARGB_8888_Config + 1
56 };
57
58 /**
59 * Default construct creates a bitmap with zero width and height, and no pixels.
60 * Its config is set to kNo_Config.
61 */
62 SkBitmap();
63
64 /**
65 * Copy the settings from the src into this bitmap. If the src has pixels
66 * allocated, they will be shared, not copied, so that the two bitmaps will
67 * reference the same memory for the pixels. If a deep copy is needed,
68 * where the new bitmap has its own separate copy of the pixels, use
69 * deepCopyTo().
70 */
71 SkBitmap(const SkBitmap& src);
72
73 ~SkBitmap();
74
75 /** Copies the src bitmap into this bitmap. Ownership of the src bitmap's pixels remains
76 with the src bitmap.
77 */
78 SkBitmap& operator=(const SkBitmap& src);
79 /** Swap the fields of the two bitmaps. This routine is guaranteed to never fail or throw.
80 */
81 // This method is not exported to java.
82 void swap(SkBitmap& other);
83
84 /** Return true iff the bitmap has empty dimensions.
85 */
empty()86 bool empty() const { return 0 == fWidth || 0 == fHeight; }
87
88 /** Return true iff the bitmap has no pixelref. Note: this can return true even if the
89 dimensions of the bitmap are > 0 (see empty()).
90 */
isNull()91 bool isNull() const { return NULL == fPixelRef; }
92
93 /** Return the config for the bitmap. */
config()94 Config config() const { return (Config)fConfig; }
95
96 SK_ATTR_DEPRECATED("use config()")
getConfig()97 Config getConfig() const { return this->config(); }
98
99 /** Return the bitmap's width, in pixels. */
width()100 int width() const { return fWidth; }
101
102 /** Return the bitmap's height, in pixels. */
height()103 int height() const { return fHeight; }
104
105 /** Return the number of bytes between subsequent rows of the bitmap. */
rowBytes()106 size_t rowBytes() const { return fRowBytes; }
107
108 /** Return the shift amount per pixel (i.e. 0 for 1-byte per pixel, 1 for
109 2-bytes per pixel configs, 2 for 4-bytes per pixel configs). Return 0
110 for configs that are not at least 1-byte per pixel (e.g. kA1_Config
111 or kNo_Config)
112 */
shiftPerPixel()113 int shiftPerPixel() const { return fBytesPerPixel >> 1; }
114
115 /** Return the number of bytes per pixel based on the config. If the config
116 does not have at least 1 byte per (e.g. kA1_Config) then 0 is returned.
117 */
bytesPerPixel()118 int bytesPerPixel() const { return fBytesPerPixel; }
119
120 /** Return the rowbytes expressed as a number of pixels (like width and
121 height). Note, for 1-byte per pixel configs like kA8_Config, this will
122 return the same as rowBytes(). Is undefined for configs that are less
123 than 1-byte per pixel (e.g. kA1_Config)
124 */
rowBytesAsPixels()125 int rowBytesAsPixels() const { return fRowBytes >> (fBytesPerPixel >> 1); }
126
alphaType()127 SkAlphaType alphaType() const { return (SkAlphaType)fAlphaType; }
128
129 /**
130 * Set the bitmap's alphaType, returning true on success. If false is
131 * returned, then the specified new alphaType is incompatible with the
132 * Config, and the current alphaType is unchanged.
133 */
134 bool setAlphaType(SkAlphaType);
135
136 /** Return the address of the pixels for this SkBitmap.
137 */
getPixels()138 void* getPixels() const { return fPixels; }
139
140 /** Return the byte size of the pixels, based on the height and rowBytes.
141 Note this truncates the result to 32bits. Call getSize64() to detect
142 if the real size exceeds 32bits.
143 */
getSize()144 size_t getSize() const { return fHeight * fRowBytes; }
145
146 /** Return the number of bytes from the pointer returned by getPixels()
147 to the end of the allocated space in the buffer. Required in
148 cases where extractSubset has been called.
149 */
150 size_t getSafeSize() const ;
151
152 /** Return the byte size of the pixels, based on the height and rowBytes.
153 This routine is slightly slower than getSize(), but does not truncate
154 the answer to 32bits.
155 */
getSize64()156 Sk64 getSize64() const {
157 Sk64 size;
158 size.setMul(fHeight, fRowBytes);
159 return size;
160 }
161
162 /** Same as getSafeSize(), but does not truncate the answer to 32bits.
163 */
164 Sk64 getSafeSize64() const ;
165
166 /** Returns true if this bitmap is marked as immutable, meaning that the
167 contents of its pixels will not change for the lifetime of the bitmap.
168 */
169 bool isImmutable() const;
170
171 /** Marks this bitmap as immutable, meaning that the contents of its
172 pixels will not change for the lifetime of the bitmap and of the
173 underlying pixelref. This state can be set, but it cannot be
174 cleared once it is set. This state propagates to all other bitmaps
175 that share the same pixelref.
176 */
177 void setImmutable();
178
179 /** Returns true if the bitmap is opaque (has no translucent/transparent pixels).
180 */
isOpaque()181 bool isOpaque() const {
182 return SkAlphaTypeIsOpaque(this->alphaType());
183 }
184
185 /** Returns true if the bitmap is volatile (i.e. should not be cached by devices.)
186 */
187 bool isVolatile() const;
188
189 /** Specify whether this bitmap is volatile. Bitmaps are not volatile by
190 default. Temporary bitmaps that are discarded after use should be
191 marked as volatile. This provides a hint to the device that the bitmap
192 should not be cached. Providing this hint when appropriate can
193 improve performance by avoiding unnecessary overhead and resource
194 consumption on the device.
195 */
196 void setIsVolatile(bool);
197
198 /** Reset the bitmap to its initial state (see default constructor). If we are a (shared)
199 owner of the pixels, that ownership is decremented.
200 */
201 void reset();
202
203 /** Given a config and a width, this computes the optimal rowBytes value. This is called automatically
204 if you pass 0 for rowBytes to setConfig().
205 */
206 static size_t ComputeRowBytes(Config c, int width);
207
208 /** Return the bytes-per-pixel for the specified config. If the config is
209 not at least 1-byte per pixel, return 0, including for kNo_Config.
210 */
211 static int ComputeBytesPerPixel(Config c);
212
213 /** Return the shift-per-pixel for the specified config. If the config is
214 not at least 1-byte per pixel, return 0, including for kNo_Config.
215 */
ComputeShiftPerPixel(Config c)216 static int ComputeShiftPerPixel(Config c) {
217 return ComputeBytesPerPixel(c) >> 1;
218 }
219
220 static Sk64 ComputeSize64(Config, int width, int height);
221 static size_t ComputeSize(Config, int width, int height);
222
223 /**
224 * This will brute-force return true if all of the pixels in the bitmap
225 * are opaque. If it fails to read the pixels, or encounters an error,
226 * it will return false.
227 *
228 * Since this can be an expensive operation, the bitmap stores a flag for
229 * this (isOpaque). Only call this if you need to compute this value from
230 * "unknown" pixels.
231 */
232 static bool ComputeIsOpaque(const SkBitmap&);
233
234 /**
235 * Return the bitmap's bounds [0, 0, width, height] as an SkRect
236 */
237 void getBounds(SkRect* bounds) const;
238 void getBounds(SkIRect* bounds) const;
239
240 /** Set the bitmap's config and dimensions. If rowBytes is 0, then
241 ComputeRowBytes() is called to compute the optimal value. This resets
242 any pixel/colortable ownership, just like reset().
243 */
244 bool setConfig(Config, int width, int height, size_t rowBytes, SkAlphaType);
245
246 bool setConfig(Config config, int width, int height, size_t rowBytes = 0) {
247 return this->setConfig(config, width, height, rowBytes,
248 kPremul_SkAlphaType);
249 }
250
251 bool setConfig(const SkImageInfo& info, size_t rowBytes = 0);
252
253 /**
254 * If the bitmap's config can be represented as SkImageInfo, return true,
255 * and if info is not-null, set it to the bitmap's info. If it cannot be
256 * represented as SkImageInfo, return false and ignore the info parameter.
257 */
258 bool asImageInfo(SkImageInfo* info) const;
259
260 /** Use this to assign a new pixel address for an existing bitmap. This
261 will automatically release any pixelref previously installed. Only call
262 this if you are handling ownership/lifetime of the pixel memory.
263
264 If the bitmap retains a reference to the colortable (assuming it is
265 not null) it will take care of incrementing the reference count.
266
267 @param pixels Address for the pixels, managed by the caller.
268 @param ctable ColorTable (or null) that matches the specified pixels
269 */
270 void setPixels(void* p, SkColorTable* ctable = NULL);
271
272 /** Copies the bitmap's pixels to the location pointed at by dst and returns
273 true if possible, returns false otherwise.
274
275 In the case when the dstRowBytes matches the bitmap's rowBytes, the copy
276 may be made faster by copying over the dst's per-row padding (for all
277 rows but the last). By setting preserveDstPad to true the caller can
278 disable this optimization and ensure that pixels in the padding are not
279 overwritten.
280
281 Always returns false for RLE formats.
282
283 @param dst Location of destination buffer.
284 @param dstSize Size of destination buffer. Must be large enough to hold
285 pixels using indicated stride.
286 @param dstRowBytes Width of each line in the buffer. If 0, uses
287 bitmap's internal stride.
288 @param preserveDstPad Must we preserve padding in the dst
289 */
290 bool copyPixelsTo(void* const dst, size_t dstSize, size_t dstRowBytes = 0,
291 bool preserveDstPad = false) const;
292
293 /** Use the standard HeapAllocator to create the pixelref that manages the
294 pixel memory. It will be sized based on the current width/height/config.
295 If this is called multiple times, a new pixelref object will be created
296 each time.
297
298 If the bitmap retains a reference to the colortable (assuming it is
299 not null) it will take care of incrementing the reference count.
300
301 @param ctable ColorTable (or null) to use with the pixels that will
302 be allocated. Only used if config == Index8_Config
303 @return true if the allocation succeeds. If not the pixelref field of
304 the bitmap will be unchanged.
305 */
306 bool allocPixels(SkColorTable* ctable = NULL) {
307 return this->allocPixels(NULL, ctable);
308 }
309
310 /** Use the specified Allocator to create the pixelref that manages the
311 pixel memory. It will be sized based on the current width/height/config.
312 If this is called multiple times, a new pixelref object will be created
313 each time.
314
315 If the bitmap retains a reference to the colortable (assuming it is
316 not null) it will take care of incrementing the reference count.
317
318 @param allocator The Allocator to use to create a pixelref that can
319 manage the pixel memory for the current
320 width/height/config. If allocator is NULL, the standard
321 HeapAllocator will be used.
322 @param ctable ColorTable (or null) to use with the pixels that will
323 be allocated. Only used if config == Index8_Config.
324 If it is non-null and the config is not Index8, it will
325 be ignored.
326 @return true if the allocation succeeds. If not the pixelref field of
327 the bitmap will be unchanged.
328 */
329 bool allocPixels(Allocator* allocator, SkColorTable* ctable);
330
331 /** Return the current pixelref object, if any
332 */
pixelRef()333 SkPixelRef* pixelRef() const { return fPixelRef; }
334 /** Return the offset into the pixelref, if any. Will return 0 if there is
335 no pixelref installed.
336 */
pixelRefOffset()337 size_t pixelRefOffset() const { return fPixelRefOffset; }
338 /** Assign a pixelref and optional offset. Pixelrefs are reference counted,
339 so the existing one (if any) will be unref'd and the new one will be
340 ref'd.
341 */
342 SkPixelRef* setPixelRef(SkPixelRef* pr, size_t offset = 0);
343
344 /** Call this to ensure that the bitmap points to the current pixel address
345 in the pixelref. Balance it with a call to unlockPixels(). These calls
346 are harmless if there is no pixelref.
347 */
348 void lockPixels() const;
349 /** When you are finished access the pixel memory, call this to balance a
350 previous call to lockPixels(). This allows pixelrefs that implement
351 cached/deferred image decoding to know when there are active clients of
352 a given image.
353 */
354 void unlockPixels() const;
355
356 /**
357 * Some bitmaps can return a copy of their pixels for lockPixels(), but
358 * that copy, if modified, will not be pushed back. These bitmaps should
359 * not be used as targets for a raster device/canvas (since all pixels
360 * modifications will be lost when unlockPixels() is called.)
361 */
362 bool lockPixelsAreWritable() const;
363
364 /** Call this to be sure that the bitmap is valid enough to be drawn (i.e.
365 it has non-null pixels, and if required by its config, it has a
366 non-null colortable. Returns true if all of the above are met.
367 */
readyToDraw()368 bool readyToDraw() const {
369 return this->getPixels() != NULL &&
370 (this->config() != kIndex8_Config || NULL != fColorTable);
371 }
372
373 /** Returns the pixelRef's texture, or NULL
374 */
375 GrTexture* getTexture() const;
376
377 /** Return the bitmap's colortable, if it uses one (i.e. fConfig is
378 kIndex8_Config) and the pixels are locked.
379 Otherwise returns NULL. Does not affect the colortable's
380 reference count.
381 */
getColorTable()382 SkColorTable* getColorTable() const { return fColorTable; }
383
384 /** Returns a non-zero, unique value corresponding to the pixels in our
385 pixelref. Each time the pixels are changed (and notifyPixelsChanged
386 is called), a different generation ID will be returned. Finally, if
387 their is no pixelRef then zero is returned.
388 */
389 uint32_t getGenerationID() const;
390
391 /** Call this if you have changed the contents of the pixels. This will in-
392 turn cause a different generation ID value to be returned from
393 getGenerationID().
394 */
395 void notifyPixelsChanged() const;
396
397 /**
398 * Fill the entire bitmap with the specified color.
399 * If the bitmap's config does not support alpha (e.g. 565) then the alpha
400 * of the color is ignored (treated as opaque). If the config only supports
401 * alpha (e.g. A1 or A8) then the color's r,g,b components are ignored.
402 */
eraseColor(SkColor c)403 void eraseColor(SkColor c) const {
404 this->eraseARGB(SkColorGetA(c), SkColorGetR(c), SkColorGetG(c),
405 SkColorGetB(c));
406 }
407
408 /**
409 * Fill the entire bitmap with the specified color.
410 * If the bitmap's config does not support alpha (e.g. 565) then the alpha
411 * of the color is ignored (treated as opaque). If the config only supports
412 * alpha (e.g. A1 or A8) then the color's r,g,b components are ignored.
413 */
414 void eraseARGB(U8CPU a, U8CPU r, U8CPU g, U8CPU b) const;
415
416 SK_ATTR_DEPRECATED("use eraseARGB or eraseColor")
eraseRGB(U8CPU r,U8CPU g,U8CPU b)417 void eraseRGB(U8CPU r, U8CPU g, U8CPU b) const {
418 this->eraseARGB(0xFF, r, g, b);
419 }
420
421 /**
422 * Fill the specified area of this bitmap with the specified color.
423 * If the bitmap's config does not support alpha (e.g. 565) then the alpha
424 * of the color is ignored (treated as opaque). If the config only supports
425 * alpha (e.g. A1 or A8) then the color's r,g,b components are ignored.
426 */
427 void eraseArea(const SkIRect& area, SkColor c) const;
428
429 /** Scroll (a subset of) the contents of this bitmap by dx/dy. If there are
430 no pixels allocated (i.e. getPixels() returns null) the method will
431 still update the inval region (if present). If the bitmap is immutable,
432 do nothing and return false.
433
434 @param subset The subset of the bitmap to scroll/move. To scroll the
435 entire contents, specify [0, 0, width, height] or just
436 pass null.
437 @param dx The amount to scroll in X
438 @param dy The amount to scroll in Y
439 @param inval Optional (may be null). Returns the area of the bitmap that
440 was scrolled away. E.g. if dx = dy = 0, then inval would
441 be set to empty. If dx >= width or dy >= height, then
442 inval would be set to the entire bounds of the bitmap.
443 @return true if the scroll was doable. Will return false if the bitmap
444 uses an unsupported config for scrolling (only kA8,
445 kIndex8, kRGB_565, kARGB_4444, kARGB_8888 are supported).
446 If no pixels are present (i.e. getPixels() returns false)
447 inval will still be updated, and true will be returned.
448 */
449 bool scrollRect(const SkIRect* subset, int dx, int dy,
450 SkRegion* inval = NULL) const;
451
452 /**
453 * Return the SkColor of the specified pixel. In most cases this will
454 * require un-premultiplying the color. Alpha only configs (A1 and A8)
455 * return black with the appropriate alpha set. The value is undefined
456 * for kNone_Config or if x or y are out of bounds, or if the bitmap
457 * does not have any pixels (or has not be locked with lockPixels()).
458 */
459 SkColor getColor(int x, int y) const;
460
461 /** Returns the address of the specified pixel. This performs a runtime
462 check to know the size of the pixels, and will return the same answer
463 as the corresponding size-specific method (e.g. getAddr16). Since the
464 check happens at runtime, it is much slower than using a size-specific
465 version. Unlike the size-specific methods, this routine also checks if
466 getPixels() returns null, and returns that. The size-specific routines
467 perform a debugging assert that getPixels() is not null, but they do
468 not do any runtime checks.
469 */
470 void* getAddr(int x, int y) const;
471
472 /** Returns the address of the pixel specified by x,y for 32bit pixels.
473 * In debug build, this asserts that the pixels are allocated and locked,
474 * and that the config is 32-bit, however none of these checks are performed
475 * in the release build.
476 */
477 inline uint32_t* getAddr32(int x, int y) const;
478
479 /** Returns the address of the pixel specified by x,y for 16bit pixels.
480 * In debug build, this asserts that the pixels are allocated and locked,
481 * and that the config is 16-bit, however none of these checks are performed
482 * in the release build.
483 */
484 inline uint16_t* getAddr16(int x, int y) const;
485
486 /** Returns the address of the pixel specified by x,y for 8bit pixels.
487 * In debug build, this asserts that the pixels are allocated and locked,
488 * and that the config is 8-bit, however none of these checks are performed
489 * in the release build.
490 */
491 inline uint8_t* getAddr8(int x, int y) const;
492
493 /** Returns the color corresponding to the pixel specified by x,y for
494 * colortable based bitmaps.
495 * In debug build, this asserts that the pixels are allocated and locked,
496 * that the config is kIndex8, and that the colortable is allocated,
497 * however none of these checks are performed in the release build.
498 */
499 inline SkPMColor getIndex8Color(int x, int y) const;
500
501 /** Set dst to be a setset of this bitmap. If possible, it will share the
502 pixel memory, and just point into a subset of it. However, if the config
503 does not support this, a local copy will be made and associated with
504 the dst bitmap. If the subset rectangle, intersected with the bitmap's
505 dimensions is empty, or if there is an unsupported config, false will be
506 returned and dst will be untouched.
507 @param dst The bitmap that will be set to a subset of this bitmap
508 @param subset The rectangle of pixels in this bitmap that dst will
509 reference.
510 @return true if the subset copy was successfully made.
511 */
512 bool extractSubset(SkBitmap* dst, const SkIRect& subset) const;
513
514 /** Makes a deep copy of this bitmap, respecting the requested config,
515 * and allocating the dst pixels on the cpu.
516 * Returns false if either there is an error (i.e. the src does not have
517 * pixels) or the request cannot be satisfied (e.g. the src has per-pixel
518 * alpha, and the requested config does not support alpha).
519 * @param dst The bitmap to be sized and allocated
520 * @param c The desired config for dst
521 * @param allocator Allocator used to allocate the pixelref for the dst
522 * bitmap. If this is null, the standard HeapAllocator
523 * will be used.
524 * @return true if the copy could be made.
525 */
526 bool copyTo(SkBitmap* dst, Config c, Allocator* allocator = NULL) const;
527
528 /** Makes a deep copy of this bitmap, respecting the requested config, and
529 * with custom allocation logic that will keep the copied pixels
530 * in the same domain as the source: If the src pixels are allocated for
531 * the cpu, then so will the dst. If the src pixels are allocated on the
532 * gpu (typically as a texture), the it will do the same for the dst.
533 * If the request cannot be fulfilled, returns false and dst is unmodified.
534 */
535 bool deepCopyTo(SkBitmap* dst, Config c) const;
536
537 /** Returns true if this bitmap can be deep copied into the requested config
538 by calling copyTo().
539 */
540 bool canCopyTo(Config newConfig) const;
541
542 SK_ATTR_DEPRECATED("use setFilterLevel on SkPaint")
543 void buildMipMap(bool forceRebuild = false);
544
545 #ifdef SK_BUILD_FOR_ANDROID
hasHardwareMipMap()546 bool hasHardwareMipMap() const {
547 return (fFlags & kHasHardwareMipMap_Flag) != 0;
548 }
549
setHasHardwareMipMap(bool hasHardwareMipMap)550 void setHasHardwareMipMap(bool hasHardwareMipMap) {
551 if (hasHardwareMipMap) {
552 fFlags |= kHasHardwareMipMap_Flag;
553 } else {
554 fFlags &= ~kHasHardwareMipMap_Flag;
555 }
556 }
557 #endif
558
extractAlpha(SkBitmap * dst)559 bool extractAlpha(SkBitmap* dst) const {
560 return this->extractAlpha(dst, NULL, NULL, NULL);
561 }
562
extractAlpha(SkBitmap * dst,const SkPaint * paint,SkIPoint * offset)563 bool extractAlpha(SkBitmap* dst, const SkPaint* paint,
564 SkIPoint* offset) const {
565 return this->extractAlpha(dst, paint, NULL, offset);
566 }
567
568 /** Set dst to contain alpha layer of this bitmap. If destination bitmap
569 fails to be initialized, e.g. because allocator can't allocate pixels
570 for it, dst will not be modified and false will be returned.
571
572 @param dst The bitmap to be filled with alpha layer
573 @param paint The paint to draw with
574 @param allocator Allocator used to allocate the pixelref for the dst
575 bitmap. If this is null, the standard HeapAllocator
576 will be used.
577 @param offset If not null, it is set to top-left coordinate to position
578 the returned bitmap so that it visually lines up with the
579 original
580 */
581 bool extractAlpha(SkBitmap* dst, const SkPaint* paint, Allocator* allocator,
582 SkIPoint* offset) const;
583
584 /** The following two functions provide the means to both flatten and
585 unflatten the bitmap AND its pixels into the provided buffer.
586 It is recommended that you do not call these functions directly,
587 but instead call the write/readBitmap functions on the respective
588 buffers as they can optimize the recording process and avoid recording
589 duplicate bitmaps and pixelRefs.
590 */
591 void flatten(SkFlattenableWriteBuffer&) const;
592 void unflatten(SkFlattenableReadBuffer&);
593
SkDEBUGCODE(void validate ()const;)594 SkDEBUGCODE(void validate() const;)
595
596 class Allocator : public SkRefCnt {
597 public:
598 SK_DECLARE_INST_COUNT(Allocator)
599
600 /** Allocate the pixel memory for the bitmap, given its dimensions and
601 config. Return true on success, where success means either setPixels
602 or setPixelRef was called. The pixels need not be locked when this
603 returns. If the config requires a colortable, it also must be
604 installed via setColorTable. If false is returned, the bitmap and
605 colortable should be left unchanged.
606 */
607 virtual bool allocPixelRef(SkBitmap*, SkColorTable*) = 0;
608 private:
609 typedef SkRefCnt INHERITED;
610 };
611
612 /** Subclass of Allocator that returns a pixelref that allocates its pixel
613 memory from the heap. This is the default Allocator invoked by
614 allocPixels().
615 */
616 class HeapAllocator : public Allocator {
617 public:
618 virtual bool allocPixelRef(SkBitmap*, SkColorTable*);
619 };
620
621 class RLEPixels {
622 public:
623 RLEPixels(int width, int height);
624 virtual ~RLEPixels();
625
packedAtY(int y)626 uint8_t* packedAtY(int y) const {
627 SkASSERT((unsigned)y < (unsigned)fHeight);
628 return fYPtrs[y];
629 }
630
631 // called by subclasses during creation
setPackedAtY(int y,uint8_t * addr)632 void setPackedAtY(int y, uint8_t* addr) {
633 SkASSERT((unsigned)y < (unsigned)fHeight);
634 fYPtrs[y] = addr;
635 }
636
637 private:
638 uint8_t** fYPtrs;
639 int fHeight;
640 };
641
642 SkDEVCODE(void toString(SkString* str) const;)
643
644 private:
645 struct MipMap;
646 mutable MipMap* fMipMap;
647
648 mutable SkPixelRef* fPixelRef;
649 mutable size_t fPixelRefOffset;
650 mutable int fPixelLockCount;
651 // either user-specified (in which case it is not treated as mutable)
652 // or a cache of the returned value from fPixelRef->lockPixels()
653 mutable void* fPixels;
654 mutable SkColorTable* fColorTable; // only meaningful for kIndex8
655
656 enum Flags {
657 kImageIsOpaque_Flag = 0x01,
658 kImageIsVolatile_Flag = 0x02,
659 kImageIsImmutable_Flag = 0x04,
660 #ifdef SK_BUILD_FOR_ANDROID
661 /* A hint for the renderer responsible for drawing this bitmap
662 * indicating that it should attempt to use mipmaps when this bitmap
663 * is drawn scaled down.
664 */
665 kHasHardwareMipMap_Flag = 0x08,
666 #endif
667 };
668
669 uint32_t fRowBytes;
670 uint32_t fWidth;
671 uint32_t fHeight;
672 uint8_t fConfig;
673 uint8_t fAlphaType;
674 uint8_t fFlags;
675 uint8_t fBytesPerPixel; // based on config
676
677 void internalErase(const SkIRect&, U8CPU a, U8CPU r, U8CPU g, U8CPU b)const;
678
679 /* Internal computations for safe size.
680 */
681 static Sk64 ComputeSafeSize64(Config config,
682 uint32_t width,
683 uint32_t height,
684 size_t rowBytes);
685 static size_t ComputeSafeSize(Config config,
686 uint32_t width,
687 uint32_t height,
688 size_t rowBytes);
689
690 /* Unreference any pixelrefs or colortables
691 */
692 void freePixels();
693 void updatePixelsFromRef() const;
694
695 static SkFixed ComputeMipLevel(SkFixed sx, SkFixed dy);
696
697 /** Given scale factors sx, sy, determine the miplevel available in the
698 bitmap, and return it (this is the amount to shift matrix iterators
699 by). If dst is not null, it is set to the correct level.
700 */
701 int extractMipLevel(SkBitmap* dst, SkFixed sx, SkFixed sy);
702 bool hasMipMap() const;
703 void freeMipMap();
704
705 friend struct SkBitmapProcState;
706 };
707
708 class SkAutoLockPixels : public SkNoncopyable {
709 public:
fBitmap(bm)710 SkAutoLockPixels(const SkBitmap& bm, bool doLock = true) : fBitmap(bm) {
711 fDidLock = doLock;
712 if (doLock) {
713 bm.lockPixels();
714 }
715 }
~SkAutoLockPixels()716 ~SkAutoLockPixels() {
717 if (fDidLock) {
718 fBitmap.unlockPixels();
719 }
720 }
721
722 private:
723 const SkBitmap& fBitmap;
724 bool fDidLock;
725 };
726 //TODO(mtklein): uncomment when 71713004 lands and Chromium's fixed.
727 //#define SkAutoLockPixels(...) SK_REQUIRE_LOCAL_VAR(SkAutoLockPixels)
728
729 /** Helper class that performs the lock/unlockColors calls on a colortable.
730 The destructor will call unlockColors(false) if it has a bitmap's colortable
731 */
732 class SkAutoLockColors : public SkNoncopyable {
733 public:
734 /** Initialize with no bitmap. Call lockColors(bitmap) to lock bitmap's
735 colortable
736 */
SkAutoLockColors()737 SkAutoLockColors() : fCTable(NULL), fColors(NULL) {}
738 /** Initialize with bitmap, locking its colortable if present
739 */
SkAutoLockColors(const SkBitmap & bm)740 explicit SkAutoLockColors(const SkBitmap& bm) {
741 fCTable = bm.getColorTable();
742 fColors = fCTable ? fCTable->lockColors() : NULL;
743 }
744 /** Initialize with a colortable (may be null)
745 */
SkAutoLockColors(SkColorTable * ctable)746 explicit SkAutoLockColors(SkColorTable* ctable) {
747 fCTable = ctable;
748 fColors = ctable ? ctable->lockColors() : NULL;
749 }
~SkAutoLockColors()750 ~SkAutoLockColors() {
751 if (fCTable) {
752 fCTable->unlockColors();
753 }
754 }
755
756 /** Return the currently locked colors, or NULL if no bitmap's colortable
757 is currently locked.
758 */
colors()759 const SkPMColor* colors() const { return fColors; }
760
761 /** Locks the table and returns is colors (assuming ctable is not null) and
762 unlocks the previous table if one was present
763 */
lockColors(SkColorTable * ctable)764 const SkPMColor* lockColors(SkColorTable* ctable) {
765 if (fCTable) {
766 fCTable->unlockColors();
767 }
768 fCTable = ctable;
769 fColors = ctable ? ctable->lockColors() : NULL;
770 return fColors;
771 }
772
lockColors(const SkBitmap & bm)773 const SkPMColor* lockColors(const SkBitmap& bm) {
774 return this->lockColors(bm.getColorTable());
775 }
776
777 private:
778 SkColorTable* fCTable;
779 const SkPMColor* fColors;
780 };
781 #define SkAutoLockColors(...) SK_REQUIRE_LOCAL_VAR(SkAutoLockColors)
782
783 ///////////////////////////////////////////////////////////////////////////////
784
getAddr32(int x,int y)785 inline uint32_t* SkBitmap::getAddr32(int x, int y) const {
786 SkASSERT(fPixels);
787 SkASSERT(fConfig == kARGB_8888_Config);
788 SkASSERT((unsigned)x < fWidth && (unsigned)y < fHeight);
789 return (uint32_t*)((char*)fPixels + y * fRowBytes + (x << 2));
790 }
791
getAddr16(int x,int y)792 inline uint16_t* SkBitmap::getAddr16(int x, int y) const {
793 SkASSERT(fPixels);
794 SkASSERT(fConfig == kRGB_565_Config || fConfig == kARGB_4444_Config);
795 SkASSERT((unsigned)x < fWidth && (unsigned)y < fHeight);
796 return (uint16_t*)((char*)fPixels + y * fRowBytes + (x << 1));
797 }
798
getAddr8(int x,int y)799 inline uint8_t* SkBitmap::getAddr8(int x, int y) const {
800 SkASSERT(fPixels);
801 SkASSERT(fConfig == kA8_Config || fConfig == kIndex8_Config);
802 SkASSERT((unsigned)x < fWidth && (unsigned)y < fHeight);
803 return (uint8_t*)fPixels + y * fRowBytes + x;
804 }
805
getIndex8Color(int x,int y)806 inline SkPMColor SkBitmap::getIndex8Color(int x, int y) const {
807 SkASSERT(fPixels);
808 SkASSERT(fConfig == kIndex8_Config);
809 SkASSERT((unsigned)x < fWidth && (unsigned)y < fHeight);
810 SkASSERT(fColorTable);
811 return (*fColorTable)[*((const uint8_t*)fPixels + y * fRowBytes + x)];
812 }
813
814 #endif
815