[chromium] Replace destRect with destOffset in texture upload
[WebKit-https.git] / Source / WebCore / platform / graphics / chromium / LayerTextureSubImage.cpp
1 /*
2  * Copyright (C) 2011 Google 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  *
8  * 1.  Redistributions of source code must retain the above copyright
9  *     notice, this list of conditions and the following disclaimer.
10  * 2.  Redistributions in binary form must reproduce the above copyright
11  *     notice, this list of conditions and the following disclaimer in the
12  *     documentation and/or other materials provided with the distribution.
13  *
14  * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
15  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
16  * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
17  * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
18  * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
19  * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
20  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
21  * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
23  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
24  */
25
26 #include "config.h"
27
28 #if USE(ACCELERATED_COMPOSITING)
29
30 #include "LayerTextureSubImage.h"
31
32 #include "Extensions3DChromium.h"
33 #include "LayerRendererChromium.h" // For GLC() macro
34 #include "TraceEvent.h"
35 #include <public/WebGraphicsContext3D.h>
36
37 using WebKit::WebGraphicsContext3D;
38
39 namespace WebCore {
40
41 LayerTextureSubImage::LayerTextureSubImage(bool useMapTexSubImage)
42     : m_useMapTexSubImage(useMapTexSubImage)
43     , m_subImageSize(0)
44 {
45 }
46
47 LayerTextureSubImage::~LayerTextureSubImage()
48 {
49 }
50
51 void LayerTextureSubImage::upload(const uint8_t* image, const IntRect& imageRect,
52                                   const IntRect& sourceRect, const IntSize& destOffset,
53                                   GC3Denum format, WebGraphicsContext3D* context)
54 {
55     if (m_useMapTexSubImage)
56         uploadWithMapTexSubImage(image, imageRect, sourceRect, destOffset, format, context);
57     else
58         uploadWithTexSubImage(image, imageRect, sourceRect, destOffset, format, context);
59 }
60
61 void LayerTextureSubImage::uploadWithTexSubImage(const uint8_t* image, const IntRect& imageRect,
62                                                  const IntRect& sourceRect, const IntSize& destOffset,
63                                                  GC3Denum format, WebGraphicsContext3D* context)
64 {
65     TRACE_EVENT0("cc", "LayerTextureSubImage::uploadWithTexSubImage");
66
67     // Offset from image-rect to source-rect.
68     IntPoint offset(sourceRect.x() - imageRect.x(), sourceRect.y() - imageRect.y());
69
70     const uint8_t* pixelSource;
71     if (imageRect.width() == sourceRect.width() && !offset.x())
72         pixelSource = &image[4 * offset.y() * imageRect.width()];
73     else {
74         size_t neededSize = 4 * sourceRect.width() * sourceRect.height();
75         if (m_subImageSize < neededSize) {
76           m_subImage = adoptArrayPtr(new uint8_t[neededSize]);
77           m_subImageSize = neededSize;
78         }
79         // Strides not equal, so do a row-by-row memcpy from the
80         // paint results into a temp buffer for uploading.
81         for (int row = 0; row < sourceRect.height(); ++row)
82             memcpy(&m_subImage[sourceRect.width() * 4 * row],
83                    &image[4 * (offset.x() + (offset.y() + row) * imageRect.width())],
84                    sourceRect.width() * 4);
85
86         pixelSource = &m_subImage[0];
87     }
88
89     GLC(context, context->texSubImage2D(GraphicsContext3D::TEXTURE_2D, 0, destOffset.width(), destOffset.height(), sourceRect.width(), sourceRect.height(), format, GraphicsContext3D::UNSIGNED_BYTE, pixelSource));
90 }
91
92 void LayerTextureSubImage::uploadWithMapTexSubImage(const uint8_t* image, const IntRect& imageRect,
93                                                     const IntRect& sourceRect, const IntSize& destOffset,
94                                                     GC3Denum format, WebGraphicsContext3D* context)
95 {
96     TRACE_EVENT0("cc", "LayerTextureSubImage::uploadWithMapTexSubImage");
97     // Offset from image-rect to source-rect.
98     IntPoint offset(sourceRect.x() - imageRect.x(), sourceRect.y() - imageRect.y());
99
100     // Upload tile data via a mapped transfer buffer
101     uint8_t* pixelDest = static_cast<uint8_t*>(context->mapTexSubImage2DCHROMIUM(GraphicsContext3D::TEXTURE_2D, 0, destOffset.width(), destOffset.height(), sourceRect.width(), sourceRect.height(), format, GraphicsContext3D::UNSIGNED_BYTE, Extensions3DChromium::WRITE_ONLY));
102
103     if (!pixelDest) {
104         uploadWithTexSubImage(image, imageRect, sourceRect, destOffset, format, context);
105         return;
106     }
107
108     unsigned int componentsPerPixel;
109     unsigned int bytesPerComponent;
110     if (!GraphicsContext3D::computeFormatAndTypeParameters(format, GraphicsContext3D::UNSIGNED_BYTE, &componentsPerPixel, &bytesPerComponent)) {
111         ASSERT_NOT_REACHED();
112         return;
113     }
114
115     if (imageRect.width() == sourceRect.width() && !offset.x())
116         memcpy(pixelDest, &image[offset.y() * imageRect.width() * componentsPerPixel * bytesPerComponent], imageRect.width() * sourceRect.height() * componentsPerPixel * bytesPerComponent);
117     else {
118         // Strides not equal, so do a row-by-row memcpy from the
119         // paint results into the pixelDest
120         for (int row = 0; row < sourceRect.height(); ++row)
121             memcpy(&pixelDest[sourceRect.width() * row * componentsPerPixel * bytesPerComponent],
122                    &image[4 * (offset.x() + (offset.y() + row) * imageRect.width())],
123                    sourceRect.width() * componentsPerPixel * bytesPerComponent);
124     }
125     GLC(context, context->unmapTexSubImage2DCHROMIUM(pixelDest));
126 }
127
128 } // namespace WebCore
129
130 #endif // USE(ACCELERATED_COMPOSITING)