blob: e5f7c69cb86467329fffc0bee7e1ad40fe939976 [file] [log] [blame]
epoger@google.comec3ed6a2011-07-28 14:26:00 +00001
2/*
3 * Copyright 2011 Google Inc.
4 *
5 * Use of this source code is governed by a BSD-style license that can be
6 * found in the LICENSE file.
7 */
reed@android.com0d55f1e2008-12-18 19:26:11 +00008#include "SkCGUtils.h"
reed@android.com758b1292008-12-18 17:54:12 +00009#include "SkBitmap.h"
reed@android.coma545a552009-06-29 17:07:19 +000010#include "SkColorPriv.h"
reed@android.com758b1292008-12-18 17:54:12 +000011
reed@android.com2b26cac2008-12-22 02:33:11 +000012static void SkBitmap_ReleaseInfo(void* info, const void* pixelData, size_t size) {
reed@android.com758b1292008-12-18 17:54:12 +000013 SkBitmap* bitmap = reinterpret_cast<SkBitmap*>(info);
14 delete bitmap;
15}
16
reed@android.coma545a552009-06-29 17:07:19 +000017#define HAS_ARGB_SHIFTS(a, r, g, b) \
18 (SK_A32_SHIFT == (a) && SK_R32_SHIFT == (r) \
19 && SK_G32_SHIFT == (g) && SK_B32_SHIFT == (b))
20
reed@google.com9c16bc02011-06-28 21:52:34 +000021static bool getBitmapInfo(const SkBitmap& bm,
22 size_t* bitsPerComponent,
23 CGBitmapInfo* info,
24 bool* upscaleTo32) {
25 if (upscaleTo32) {
26 *upscaleTo32 = false;
27 }
28
reed@android.com758b1292008-12-18 17:54:12 +000029 switch (bm.config()) {
reed@android.com32a42492009-07-10 03:33:52 +000030 case SkBitmap::kRGB_565_Config:
reed@google.com9c16bc02011-06-28 21:52:34 +000031 if (upscaleTo32) {
32 *upscaleTo32 = true;
33 }
reed@android.com32a42492009-07-10 03:33:52 +000034 // fall through
reed@android.com758b1292008-12-18 17:54:12 +000035 case SkBitmap::kARGB_8888_Config:
36 *bitsPerComponent = 8;
reed@android.com8ede4922009-06-22 20:04:33 +000037#if defined(SK_CPU_LENDIAN) && HAS_ARGB_SHIFTS(24, 0, 8, 16) \
reed@google.com9c16bc02011-06-28 21:52:34 +000038|| defined(SK_CPU_BENDIAN) && HAS_ARGB_SHIFTS(0, 24, 16, 8)
reed@google.com62f46592011-01-05 15:49:43 +000039 *info = kCGBitmapByteOrder32Big | kCGImageAlphaPremultipliedLast;
reed@android.com8ede4922009-06-22 20:04:33 +000040#elif defined(SK_CPU_LENDIAN) && HAS_ARGB_SHIFTS(24, 16, 8, 0) \
reed@google.com9c16bc02011-06-28 21:52:34 +000041|| defined(SK_CPU_BENDIAN) && HAS_ARGB_SHIFTS(24, 16, 8, 0)
reed@android.com8ede4922009-06-22 20:04:33 +000042 // Matches the CGBitmapInfo that Apple recommends for best
43 // performance, used by google chrome.
reed@google.com62f46592011-01-05 15:49:43 +000044 *info = kCGBitmapByteOrder32Little | kCGImageAlphaPremultipliedFirst;
reed@android.com8ede4922009-06-22 20:04:33 +000045#else
reed@google.com9c16bc02011-06-28 21:52:34 +000046 // ...add more formats as required...
reed@android.com8ede4922009-06-22 20:04:33 +000047#warning Cannot convert SkBitmap to CGImageRef with these shiftmasks. \
reed@google.com9c16bc02011-06-28 21:52:34 +000048This will probably not work.
reed@android.com8ede4922009-06-22 20:04:33 +000049 // Legacy behavior. Perhaps turn this into an error at some
50 // point.
reed@google.com62f46592011-01-05 15:49:43 +000051 *info = kCGBitmapByteOrder32Big | kCGImageAlphaPremultipliedLast;
reed@android.com8ede4922009-06-22 20:04:33 +000052#endif
reed@android.com758b1292008-12-18 17:54:12 +000053 break;
reed@android.com32a42492009-07-10 03:33:52 +000054#if 0
reed@android.com0680d6c2008-12-19 19:46:15 +000055 case SkBitmap::kRGB_565_Config:
56 // doesn't see quite right. Are they thinking 1555?
57 *bitsPerComponent = 5;
58 *info = kCGBitmapByteOrder16Little;
59 break;
reed@android.com32a42492009-07-10 03:33:52 +000060#endif
reed@android.com0680d6c2008-12-19 19:46:15 +000061 case SkBitmap::kARGB_4444_Config:
62 *bitsPerComponent = 4;
63 *info = kCGBitmapByteOrder16Little | kCGImageAlphaPremultipliedLast;
64 break;
reed@android.com758b1292008-12-18 17:54:12 +000065 default:
reed@google.com9c16bc02011-06-28 21:52:34 +000066 return false;
67 }
68 return true;
69}
70
71static SkBitmap* prepareForImageRef(const SkBitmap& bm,
72 size_t* bitsPerComponent,
73 CGBitmapInfo* info) {
74 bool upscaleTo32;
75 if (!getBitmapInfo(bm, bitsPerComponent, info, &upscaleTo32)) {
76 return NULL;
reed@android.com758b1292008-12-18 17:54:12 +000077 }
78
reed@android.com32a42492009-07-10 03:33:52 +000079 SkBitmap* copy;
80 if (upscaleTo32) {
81 copy = new SkBitmap;
82 // here we make a ceep copy of the pixels, since CG won't take our
83 // 565 directly
84 bm.copyTo(copy, SkBitmap::kARGB_8888_Config);
85 } else {
86 copy = new SkBitmap(bm);
87 }
88 return copy;
reed@android.com758b1292008-12-18 17:54:12 +000089}
90
reed@android.coma545a552009-06-29 17:07:19 +000091#undef HAS_ARGB_SHIFTS
92
reed@android.com38669c12011-01-03 13:48:50 +000093CGImageRef SkCreateCGImageRefWithColorspace(const SkBitmap& bm,
94 CGColorSpaceRef colorSpace) {
95 size_t bitsPerComponent SK_INIT_TO_AVOID_WARNING;
96 CGBitmapInfo info SK_INIT_TO_AVOID_WARNING;
reed@android.com758b1292008-12-18 17:54:12 +000097
98 SkBitmap* bitmap = prepareForImageRef(bm, &bitsPerComponent, &info);
99 if (NULL == bitmap) {
100 return NULL;
101 }
102
103 const int w = bitmap->width();
104 const int h = bitmap->height();
105 const size_t s = bitmap->getSize();
106
reed@android.com758b1292008-12-18 17:54:12 +0000107 // our provider "owns" the bitmap*, and will take care of deleting it
reed@android.com2b26cac2008-12-22 02:33:11 +0000108 // we initially lock it, so we can access the pixels. The bitmap will be deleted in the release
109 // proc, which will in turn unlock the pixels
110 bitmap->lockPixels();
111 CGDataProviderRef dataRef = CGDataProviderCreateWithData(bitmap, bitmap->getPixels(), s,
112 SkBitmap_ReleaseInfo);
113
reed@android.com38669c12011-01-03 13:48:50 +0000114 bool releaseColorSpace = false;
115 if (NULL == colorSpace) {
reed@google.comc280d112011-01-05 16:07:35 +0000116 colorSpace = CGColorSpaceCreateDeviceRGB();
reed@android.com38669c12011-01-03 13:48:50 +0000117 releaseColorSpace = true;
118 }
119
reed@android.com758b1292008-12-18 17:54:12 +0000120 CGImageRef ref = CGImageCreate(w, h, bitsPerComponent,
121 bitmap->bytesPerPixel() * 8,
reed@android.com38669c12011-01-03 13:48:50 +0000122 bitmap->rowBytes(), colorSpace, info, dataRef,
reed@android.com758b1292008-12-18 17:54:12 +0000123 NULL, false, kCGRenderingIntentDefault);
reed@android.com38669c12011-01-03 13:48:50 +0000124
125 if (releaseColorSpace) {
126 CGColorSpaceRelease(colorSpace);
127 }
reed@android.com758b1292008-12-18 17:54:12 +0000128 CGDataProviderRelease(dataRef);
129 return ref;
130}
131
reed@android.comf2b98d62010-12-20 18:26:13 +0000132void SkCGDrawBitmap(CGContextRef cg, const SkBitmap& bm, float x, float y) {
133 CGImageRef img = SkCreateCGImageRef(bm);
134
135 if (img) {
136 CGRect r = CGRectMake(0, 0, bm.width(), bm.height());
reed@google.com62f46592011-01-05 15:49:43 +0000137
reed@android.comf2b98d62010-12-20 18:26:13 +0000138 CGContextSaveGState(cg);
139 CGContextTranslateCTM(cg, x, r.size.height + y);
140 CGContextScaleCTM(cg, 1, -1);
reed@google.com62f46592011-01-05 15:49:43 +0000141
reed@android.comf2b98d62010-12-20 18:26:13 +0000142 CGContextDrawImage(cg, r, img);
reed@google.com62f46592011-01-05 15:49:43 +0000143
reed@android.comf2b98d62010-12-20 18:26:13 +0000144 CGContextRestoreGState(cg);
reed@google.com62f46592011-01-05 15:49:43 +0000145
reed@android.comf2b98d62010-12-20 18:26:13 +0000146 CGImageRelease(img);
147 }
148}
149
reed@google.com292ade62011-06-28 20:54:03 +0000150///////////////////////////////////////////////////////////////////////////////
reed@android.comf2b98d62010-12-20 18:26:13 +0000151
reed@google.com292ade62011-06-28 20:54:03 +0000152#include "SkStream.h"
153
154class SkAutoPDFRelease {
155public:
156 SkAutoPDFRelease(CGPDFDocumentRef doc) : fDoc(doc) {}
157 ~SkAutoPDFRelease() {
158 if (fDoc) {
159 CGPDFDocumentRelease(fDoc);
160 }
161 }
162private:
163 CGPDFDocumentRef fDoc;
164};
165
166static void CGDataProviderReleaseData_FromMalloc(void*, const void* data,
167 size_t size) {
168 sk_free((void*)data);
169}
170
171bool SkPDFDocumentToBitmap(SkStream* stream, SkBitmap* output) {
172 size_t size = stream->getLength();
173 void* ptr = sk_malloc_throw(size);
174 stream->read(ptr, size);
175 CGDataProviderRef data = CGDataProviderCreateWithData(NULL, ptr, size,
176 CGDataProviderReleaseData_FromMalloc);
177 if (NULL == data) {
178 return false;
179 }
180
181 CGPDFDocumentRef pdf = CGPDFDocumentCreateWithProvider(data);
182 CGDataProviderRelease(data);
183 if (NULL == pdf) {
184 return false;
185 }
186 SkAutoPDFRelease releaseMe(pdf);
187
188 CGPDFPageRef page = CGPDFDocumentGetPage(pdf, 1);
189 if (NULL == page) {
190 return false;
191 }
192
193 CGRect bounds = CGPDFPageGetBoxRect(page, kCGPDFMediaBox);
194
195 int w = (int)CGRectGetWidth(bounds);
196 int h = (int)CGRectGetHeight(bounds);
reed@google.com9c16bc02011-06-28 21:52:34 +0000197
reed@google.com292ade62011-06-28 20:54:03 +0000198 SkBitmap bitmap;
199 bitmap.setConfig(SkBitmap::kARGB_8888_Config, w, h);
200 bitmap.allocPixels();
201 bitmap.eraseColor(SK_ColorWHITE);
reed@google.com9c16bc02011-06-28 21:52:34 +0000202
203 size_t bitsPerComponent;
204 CGBitmapInfo info;
205 getBitmapInfo(bitmap, &bitsPerComponent, &info, NULL);
reed@google.comcd88d7c2011-06-29 20:12:38 +0000206
207 CGColorSpaceRef cs = CGColorSpaceCreateDeviceRGB();
208 CGContextRef ctx = CGBitmapContextCreate(bitmap.getPixels(), w, h,
209 bitsPerComponent, bitmap.rowBytes(),
210 cs, info);
211 CGColorSpaceRelease(cs);
212
reed@google.com292ade62011-06-28 20:54:03 +0000213 if (ctx) {
214 CGContextDrawPDFPage(ctx, page);
215 CGContextRelease(ctx);
216 }
217
218 output->swap(bitmap);
219 return true;
220}
reed@android.com758b1292008-12-18 17:54:12 +0000221