Implement the HTML5 canvas tainting rules to prevent potential data leakage
[WebKit-https.git] / WebCore / html / CanvasPattern.cpp
1 /*
2  * Copyright (C) 2006 Apple Computer, Inc.  All rights reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions
6  * are met:
7  * 1. Redistributions of source code must retain the above copyright
8  *    notice, this list of conditions and the following disclaimer.
9  * 2. Redistributions in binary form must reproduce the above copyright
10  *    notice, this list of conditions and the following disclaimer in the
11  *    documentation and/or other materials provided with the distribution.
12  *
13  * THIS SOFTWARE IS PROVIDED BY APPLE COMPUTER, INC. ``AS IS'' AND ANY
14  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL APPLE COMPUTER, INC. OR
17  * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21  * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 
24  */
25
26 #include "config.h"
27 #include "CanvasPattern.h"
28
29 #include "CachedImage.h"
30 #include "ExceptionCode.h"
31 #include "FloatRect.h"
32 #include "GraphicsContext.h"
33 #include "Image.h"
34
35 namespace WebCore {
36
37 void CanvasPattern::parseRepetitionType(const String& type, bool& repeatX, bool& repeatY, ExceptionCode& ec)
38 {
39     if (type.isEmpty() || type == "repeat") {
40         repeatX = true;
41         repeatY = true;
42         ec = 0;
43         return;
44     }
45     if (type == "no-repeat") {
46         repeatX = false;
47         repeatY = false;
48         ec = 0;
49         return;
50     }
51     if (type == "repeat-x") {
52         repeatX = true;
53         repeatY = false;
54         ec = 0;
55         return;
56     }
57     if (type == "repeat-y") {
58         repeatX = false;
59         repeatY = true;
60         ec = 0;
61         return;
62     }
63     ec = SYNTAX_ERR;
64 }
65
66 #if PLATFORM(CG)
67
68 CanvasPattern::CanvasPattern(CGImageRef image, bool repeatX, bool repeatY, bool originClean)
69     : RefCounted<CanvasPattern>(0)
70     , m_platformImage(image)
71     , m_cachedImage(0)
72     , m_repeatX(repeatX)
73     , m_repeatY(repeatY)
74     , m_originClean(originClean)
75 {
76 }
77
78 #elif PLATFORM(CAIRO)
79
80 CanvasPattern::CanvasPattern(cairo_surface_t* surface, bool repeatX, bool repeatY, bool originClean)
81     : RefCounted<CanvasPattern>(0)
82     , m_platformImage(cairo_surface_reference(surface))
83     , m_cachedImage(0)
84     , m_repeatX(repeatX)
85     , m_repeatY(repeatY)
86     , m_originClean(originClean)
87 {
88 }
89
90 #endif
91
92 CanvasPattern::CanvasPattern(CachedImage* cachedImage, bool repeatX, bool repeatY, bool originClean)
93     : RefCounted<CanvasPattern>(0)
94 #if PLATFORM(CG) || PLATFORM(CAIRO)
95     , m_platformImage(0)
96 #endif
97     , m_cachedImage(cachedImage)
98     , m_repeatX(repeatX)
99     , m_repeatY(repeatY)
100     , m_originClean(originClean)
101 {
102     if (cachedImage)
103         cachedImage->ref(this);
104 }
105
106 CanvasPattern::~CanvasPattern()
107 {
108 #if PLATFORM(CAIRO)
109     if (m_platformImage)
110         cairo_surface_destroy(m_platformImage);
111 #endif
112     if (m_cachedImage)
113         m_cachedImage->deref(this);
114 }
115
116 #if PLATFORM(CG)
117
118 static void patternCallback(void* info, CGContextRef context)
119 {
120     CGImageRef platformImage = static_cast<CanvasPattern*>(info)->platformImage();
121     if (platformImage) {
122         CGRect rect = GraphicsContext(context).roundToDevicePixels(
123             FloatRect(0, 0, CGImageGetWidth(platformImage), CGImageGetHeight(platformImage)));
124         CGContextDrawImage(context, rect, platformImage);
125         return;
126     }
127
128     CachedImage* cachedImage = static_cast<CanvasPattern*>(info)->cachedImage();
129     if (!cachedImage)
130         return;
131     Image* image = cachedImage->image();
132     if (!image)
133         return;
134
135     FloatRect rect = GraphicsContext(context).roundToDevicePixels(image->rect());
136
137     if (image->getCGImageRef()) {
138         CGContextDrawImage(context, rect, image->getCGImageRef());
139         // FIXME: We should refactor this code to use the platform-independent 
140         // drawing API in all cases. Then, this didDraw call will happen 
141         // automatically, and we can remove it.
142         cachedImage->didDraw(image);
143         return;
144     }
145
146     GraphicsContext(context).drawImage(image, rect);
147 }
148
149 static void patternReleaseCallback(void* info)
150 {
151     static_cast<CanvasPattern*>(info)->deref();
152 }
153
154 CGPatternRef CanvasPattern::createPattern(const CGAffineTransform& transform)
155 {
156     CGRect rect;
157     rect.origin.x = 0;
158     rect.origin.y = 0;
159     if (m_platformImage) {
160         rect.size.width = CGImageGetWidth(m_platformImage.get());
161         rect.size.height = CGImageGetHeight(m_platformImage.get());
162     } else {
163         if (!m_cachedImage)
164             return 0;
165         Image* image = m_cachedImage->image();
166         if (!image)
167             return 0;
168         rect.size.width = image->width();
169         rect.size.height = image->height();
170     }
171
172     CGAffineTransform patternTransform =
173         CGAffineTransformTranslate(CGAffineTransformScale(transform, 1, -1), 0, -rect.size.height);
174
175     float xStep = m_repeatX ? rect.size.width : FLT_MAX;
176     // If FLT_MAX should also be used for yStep, nothing is rendered. Using fractions of FLT_MAX also
177     // result in nothing being rendered. This is not a problem with xStep.
178     // INT_MAX is almost correct, but there seems to be some number wrapping occuring making the fill
179     // pattern is not filled correctly. 
180     // So, just pick a really large number that works. 
181     float yStep = m_repeatY ? rect.size.height : (100000000.0f);
182
183     const CGPatternCallbacks patternCallbacks = { 0, patternCallback, patternReleaseCallback };
184     ref();
185     return CGPatternCreate(this, rect, patternTransform, xStep, yStep,
186         kCGPatternTilingConstantSpacing, TRUE, &patternCallbacks);
187 }
188
189 #elif PLATFORM(CAIRO)
190
191 cairo_pattern_t* CanvasPattern::createPattern(const cairo_matrix_t& m)
192 {
193     cairo_surface_t* surface = 0;
194     if (m_platformImage) {
195         surface = m_platformImage;
196     } else {
197         if (!m_cachedImage)
198             return 0;
199         Image* image = m_cachedImage->image();
200         if (!image)
201             return 0;
202         surface = image->nativeImageForCurrentFrame();
203     }
204
205     if (!surface)
206         return 0;
207
208     cairo_pattern_t* pattern = cairo_pattern_create_for_surface(surface);
209     cairo_pattern_set_matrix(pattern, &m);
210     if (m_repeatX || m_repeatY)
211         cairo_pattern_set_extend(pattern, CAIRO_EXTEND_REPEAT);
212     return pattern;
213 }
214
215 #endif
216
217 }