WebPImageDecoder: Increase image/webp decoding performance 10-20%
[WebKit-https.git] / Source / WebCore / platform / image-decoders / webp / WEBPImageDecoder.cpp
1 /*
2  * Copyright (C) 2010 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  * 3.  Neither the name of Apple Computer, Inc. ("Apple") nor the names of
14  *     its contributors may be used to endorse or promote products derived
15  *     from this software without specific prior written permission.
16  *
17  * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
18  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
19  * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
20  * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
21  * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
22  * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
23  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
24  * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
26  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27  */
28
29 #include "config.h"
30 #include "WEBPImageDecoder.h"
31
32 #if USE(WEBP)
33
34 #include "webp/decode.h"
35
36 #if CPU(BIG_ENDIAN) || CPU(MIDDLE_ENDIAN)
37 inline WEBP_CSP_MODE outputMode() { return MODE_RGBA; }
38 #else // LITTLE_ENDIAN, output BGRA pixels.
39 inline WEBP_CSP_MODE outputMode() { return MODE_BGRA; }
40 #endif
41
42 namespace WebCore {
43
44 WEBPImageDecoder::WEBPImageDecoder(ImageSource::AlphaOption alphaOption,
45                                    ImageSource::GammaAndColorProfileOption gammaAndColorProfileOption)
46     : ImageDecoder(alphaOption, gammaAndColorProfileOption)
47     , m_decoder(0)
48 {
49 }
50
51 WEBPImageDecoder::~WEBPImageDecoder()
52 {
53     if (m_decoder)
54         WebPIDelete(m_decoder);
55     m_decoder = 0;
56 }
57
58 bool WEBPImageDecoder::isSizeAvailable()
59 {
60     if (!ImageDecoder::isSizeAvailable())
61          decode(true);
62
63     return ImageDecoder::isSizeAvailable();
64 }
65
66 ImageFrame* WEBPImageDecoder::frameBufferAtIndex(size_t index)
67 {
68     if (index)
69         return 0;
70
71     if (m_frameBufferCache.isEmpty()) {
72         m_frameBufferCache.resize(1);
73         m_frameBufferCache[0].setPremultiplyAlpha(m_premultiplyAlpha);
74     }
75
76     ImageFrame& frame = m_frameBufferCache[0];
77     if (frame.status() != ImageFrame::FrameComplete)
78         decode(false);
79     return &frame;
80 }
81
82 bool WEBPImageDecoder::decode(bool onlySize)
83 {
84     if (failed())
85         return false;
86
87     const uint8_t* dataBytes = reinterpret_cast<const uint8_t*>(m_data->data());
88     const size_t dataSize = m_data->size();
89
90     if (!ImageDecoder::isSizeAvailable()) {
91         static const size_t imageHeaderSize = 30;
92         if (dataSize < imageHeaderSize)
93             return false;
94         int width, height;
95         if (!WebPGetInfo(dataBytes, dataSize, &width, &height))
96             return setFailed();
97         if (!setSize(width, height))
98             return setFailed();
99     }
100
101     ASSERT(ImageDecoder::isSizeAvailable());
102     if (onlySize)
103         return true;
104
105     ASSERT(!m_frameBufferCache.isEmpty());
106     ImageFrame& buffer = m_frameBufferCache[0];
107     ASSERT(buffer.status() != ImageFrame::FrameComplete);
108
109     if (buffer.status() == ImageFrame::FrameEmpty) {
110         if (!buffer.setSize(size().width(), size().height()))
111             return setFailed();
112         buffer.setStatus(ImageFrame::FramePartial);
113         buffer.setHasAlpha(false); // FIXME: webp does not support alpha yet.
114         buffer.setOriginalFrameRect(IntRect(IntPoint(), size()));
115     }
116
117     if (!m_decoder) {
118         int rowStride = size().width() * sizeof(ImageFrame::PixelData);
119         uint8_t* output = reinterpret_cast<uint8_t*>(buffer.getAddr(0, 0));
120         int outputSize = size().height() * rowStride;
121         m_decoder = WebPINewRGB(outputMode(), output, outputSize, rowStride);
122         if (!m_decoder)
123             return setFailed();
124     }
125
126     switch (WebPIUpdate(m_decoder, dataBytes, dataSize)) {
127     case VP8_STATUS_OK:
128         buffer.setStatus(ImageFrame::FrameComplete);
129         WebPIDelete(m_decoder);
130         m_decoder = 0;
131         return true;
132     case VP8_STATUS_SUSPENDED:
133         return false;
134     default:
135         WebPIDelete(m_decoder);
136         m_decoder = 0;
137         return setFailed();
138     }
139 }
140
141 } // namespace WebCore
142
143 #endif