summaryrefslogtreecommitdiffstats
path: root/base/gfx
diff options
context:
space:
mode:
authorbrettw@google.com <brettw@google.com@0039d316-1c4b-4281-b951-d872f2087c98>2008-12-03 16:22:10 +0000
committerbrettw@google.com <brettw@google.com@0039d316-1c4b-4281-b951-d872f2087c98>2008-12-03 16:22:10 +0000
commit83c9e65546312a8d70df850a82f390f190fe4413 (patch)
tree347e411c8bcd9769f6e8a50252c607e9699c5046 /base/gfx
parentbd974b6c13642d8c98496fc1fc06ddff1665b4ac (diff)
downloadchromium_src-83c9e65546312a8d70df850a82f390f190fe4413.zip
chromium_src-83c9e65546312a8d70df850a82f390f190fe4413.tar.gz
chromium_src-83c9e65546312a8d70df850a82f390f190fe4413.tar.bz2
Move convolver and image_operations from base/gfx to skia/ext. This is just
like my previous change except does no namespace renaming and doesn't touch skia_utils. Review URL: http://codereview.chromium.org/13080 git-svn-id: svn://svn.chromium.org/chrome/trunk/src@6290 0039d316-1c4b-4281-b951-d872f2087c98
Diffstat (limited to 'base/gfx')
-rw-r--r--base/gfx/base_gfx.scons2
-rw-r--r--base/gfx/convolver.cc335
-rw-r--r--base/gfx/convolver.h137
-rw-r--r--base/gfx/convolver_unittest.cc127
-rw-r--r--base/gfx/image_operations.cc362
-rw-r--r--base/gfx/image_operations.h63
-rw-r--r--base/gfx/image_operations_unittest.cc148
-rw-r--r--base/gfx/img_resize_perftest.cc70
8 files changed, 0 insertions, 1244 deletions
diff --git a/base/gfx/base_gfx.scons b/base/gfx/base_gfx.scons
index 7173340..e699344 100644
--- a/base/gfx/base_gfx.scons
+++ b/base/gfx/base_gfx.scons
@@ -25,9 +25,7 @@ if env['PLATFORM'] == 'win32':
)
input_files = [
- 'convolver.cc',
'gdi_util.cc',
- 'image_operations.cc',
'native_theme.cc',
'png_decoder.cc',
'png_encoder.cc',
diff --git a/base/gfx/convolver.cc b/base/gfx/convolver.cc
deleted file mode 100644
index fd3503f..0000000
--- a/base/gfx/convolver.cc
+++ /dev/null
@@ -1,335 +0,0 @@
-// Copyright (c) 2006-2008 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#include <algorithm>
-
-#include "base/basictypes.h"
-#include "base/gfx/convolver.h"
-#include "base/logging.h"
-
-namespace gfx {
-
-namespace {
-
-// Converts the argument to an 8-bit unsigned value by clamping to the range
-// 0-255.
-inline uint8 ClampTo8(int32 a) {
- if (static_cast<uint32>(a) < 256)
- return a; // Avoid the extra check in the common case.
- if (a < 0)
- return 0;
- return 255;
-}
-
-// Stores a list of rows in a circular buffer. The usage is you write into it
-// by calling AdvanceRow. It will keep track of which row in the buffer it
-// should use next, and the total number of rows added.
-class CircularRowBuffer {
- public:
- // The number of pixels in each row is given in |source_row_pixel_width|.
- // The maximum number of rows needed in the buffer is |max_y_filter_size|
- // (we only need to store enough rows for the biggest filter).
- //
- // We use the |first_input_row| to compute the coordinates of all of the
- // following rows returned by Advance().
- CircularRowBuffer(int dest_row_pixel_width, int max_y_filter_size,
- int first_input_row)
- : row_byte_width_(dest_row_pixel_width * 4),
- num_rows_(max_y_filter_size),
- next_row_(0),
- next_row_coordinate_(first_input_row) {
- buffer_.resize(row_byte_width_ * max_y_filter_size);
- row_addresses_.resize(num_rows_);
- }
-
- // Moves to the next row in the buffer, returning a pointer to the beginning
- // of it.
- uint8* AdvanceRow() {
- uint8* row = &buffer_[next_row_ * row_byte_width_];
- next_row_coordinate_++;
-
- // Set the pointer to the next row to use, wrapping around if necessary.
- next_row_++;
- if (next_row_ == num_rows_)
- next_row_ = 0;
- return row;
- }
-
- // Returns a pointer to an "unrolled" array of rows. These rows will start
- // at the y coordinate placed into |*first_row_index| and will continue in
- // order for the maximum number of rows in this circular buffer.
- //
- // The |first_row_index_| may be negative. This means the circular buffer
- // starts before the top of the image (it hasn't been filled yet).
- uint8* const* GetRowAddresses(int* first_row_index) {
- // Example for a 4-element circular buffer holding coords 6-9.
- // Row 0 Coord 8
- // Row 1 Coord 9
- // Row 2 Coord 6 <- next_row_ = 2, next_row_coordinate_ = 10.
- // Row 3 Coord 7
- //
- // The "next" row is also the first (lowest) coordinate. This computation
- // may yield a negative value, but that's OK, the math will work out
- // since the user of this buffer will compute the offset relative
- // to the first_row_index and the negative rows will never be used.
- *first_row_index = next_row_coordinate_ - num_rows_;
-
- int cur_row = next_row_;
- for (int i = 0; i < num_rows_; i++) {
- row_addresses_[i] = &buffer_[cur_row * row_byte_width_];
-
- // Advance to the next row, wrapping if necessary.
- cur_row++;
- if (cur_row == num_rows_)
- cur_row = 0;
- }
- return &row_addresses_[0];
- }
-
- private:
- // The buffer storing the rows. They are packed, each one row_byte_width_.
- std::vector<uint8> buffer_;
-
- // Number of bytes per row in the |buffer_|.
- int row_byte_width_;
-
- // The number of rows available in the buffer.
- int num_rows_;
-
- // The next row index we should write into. This wraps around as the
- // circular buffer is used.
- int next_row_;
-
- // The y coordinate of the |next_row_|. This is incremented each time a
- // new row is appended and does not wrap.
- int next_row_coordinate_;
-
- // Buffer used by GetRowAddresses().
- std::vector<uint8*> row_addresses_;
-};
-
-// Convolves horizontally along a single row. The row data is given in
-// |src_data| and continues for the num_values() of the filter.
-template<bool has_alpha>
-void ConvolveHorizontally(const uint8* src_data,
- const ConvolusionFilter1D& filter,
- unsigned char* out_row) {
- // Loop over each pixel on this row in the output image.
- int num_values = filter.num_values();
- for (int out_x = 0; out_x < num_values; out_x++) {
- // Get the filter that determines the current output pixel.
- int filter_offset, filter_length;
- const int16* filter_values =
- filter.FilterForValue(out_x, &filter_offset, &filter_length);
-
- // Compute the first pixel in this row that the filter affects. It will
- // touch |filter_length| pixels (4 bytes each) after this.
- const uint8* row_to_filter = &src_data[filter_offset * 4];
-
- // Apply the filter to the row to get the destination pixel in |accum|.
- int32 accum[4] = {0};
- for (int filter_x = 0; filter_x < filter_length; filter_x++) {
- int16 cur_filter = filter_values[filter_x];
- accum[0] += cur_filter * row_to_filter[filter_x * 4 + 0];
- accum[1] += cur_filter * row_to_filter[filter_x * 4 + 1];
- accum[2] += cur_filter * row_to_filter[filter_x * 4 + 2];
- if (has_alpha)
- accum[3] += cur_filter * row_to_filter[filter_x * 4 + 3];
- }
-
- // Bring this value back in range. All of the filter scaling factors
- // are in fixed point with kShiftBits bits of fractional part.
- accum[0] >>= ConvolusionFilter1D::kShiftBits;
- accum[1] >>= ConvolusionFilter1D::kShiftBits;
- accum[2] >>= ConvolusionFilter1D::kShiftBits;
- if (has_alpha)
- accum[3] >>= ConvolusionFilter1D::kShiftBits;
-
- // Store the new pixel.
- out_row[out_x * 4 + 0] = ClampTo8(accum[0]);
- out_row[out_x * 4 + 1] = ClampTo8(accum[1]);
- out_row[out_x * 4 + 2] = ClampTo8(accum[2]);
- if (has_alpha)
- out_row[out_x * 4 + 3] = ClampTo8(accum[3]);
- }
-}
-
-// Does vertical convolusion to produce one output row. The filter values and
-// length are given in the first two parameters. These are applied to each
-// of the rows pointed to in the |source_data_rows| array, with each row
-// being |pixel_width| wide.
-//
-// The output must have room for |pixel_width * 4| bytes.
-template<bool has_alpha>
-void ConvolveVertically(const int16* filter_values,
- int filter_length,
- uint8* const* source_data_rows,
- int pixel_width,
- uint8* out_row) {
- // We go through each column in the output and do a vertical convolusion,
- // generating one output pixel each time.
- for (int out_x = 0; out_x < pixel_width; out_x++) {
- // Compute the number of bytes over in each row that the current column
- // we're convolving starts at. The pixel will cover the next 4 bytes.
- int byte_offset = out_x * 4;
-
- // Apply the filter to one column of pixels.
- int32 accum[4] = {0};
- for (int filter_y = 0; filter_y < filter_length; filter_y++) {
- int16 cur_filter = filter_values[filter_y];
- accum[0] += cur_filter * source_data_rows[filter_y][byte_offset + 0];
- accum[1] += cur_filter * source_data_rows[filter_y][byte_offset + 1];
- accum[2] += cur_filter * source_data_rows[filter_y][byte_offset + 2];
- if (has_alpha)
- accum[3] += cur_filter * source_data_rows[filter_y][byte_offset + 3];
- }
-
- // Bring this value back in range. All of the filter scaling factors
- // are in fixed point with kShiftBits bits of precision.
- accum[0] >>= ConvolusionFilter1D::kShiftBits;
- accum[1] >>= ConvolusionFilter1D::kShiftBits;
- accum[2] >>= ConvolusionFilter1D::kShiftBits;
- if (has_alpha)
- accum[3] >>= ConvolusionFilter1D::kShiftBits;
-
- // Store the new pixel.
- out_row[byte_offset + 0] = ClampTo8(accum[0]);
- out_row[byte_offset + 1] = ClampTo8(accum[1]);
- out_row[byte_offset + 2] = ClampTo8(accum[2]);
- if (has_alpha) {
- uint8 alpha = ClampTo8(accum[3]);
-
- // Make sure the alpha channel doesn't come out larger than any of the
- // color channels. We use premultipled alpha channels, so this should
- // never happen, but rounding errors will cause this from time to time.
- // These "impossible" colors will cause overflows (and hence random pixel
- // values) when the resulting bitmap is drawn to the screen.
- //
- // We only need to do this when generating the final output row (here).
- int max_color_channel = std::max(out_row[byte_offset + 0],
- std::max(out_row[byte_offset + 1], out_row[byte_offset + 2]));
- if (alpha < max_color_channel)
- out_row[byte_offset + 3] = max_color_channel;
- else
- out_row[byte_offset + 3] = alpha;
- } else {
- // No alpha channel, the image is opaque.
- out_row[byte_offset + 3] = 0xff;
- }
- }
-}
-
-} // namespace
-
-// ConvolusionFilter1D ---------------------------------------------------------
-
-void ConvolusionFilter1D::AddFilter(int filter_offset,
- const float* filter_values,
- int filter_length) {
- FilterInstance instance;
- instance.data_location = static_cast<int>(filter_values_.size());
- instance.offset = filter_offset;
- instance.length = filter_length;
- filters_.push_back(instance);
-
- DCHECK(filter_length > 0);
- for (int i = 0; i < filter_length; i++)
- filter_values_.push_back(FloatToFixed(filter_values[i]));
-
- max_filter_ = std::max(max_filter_, filter_length);
-}
-
-void ConvolusionFilter1D::AddFilter(int filter_offset,
- const int16* filter_values,
- int filter_length) {
- FilterInstance instance;
- instance.data_location = static_cast<int>(filter_values_.size());
- instance.offset = filter_offset;
- instance.length = filter_length;
- filters_.push_back(instance);
-
- DCHECK(filter_length > 0);
- for (int i = 0; i < filter_length; i++)
- filter_values_.push_back(filter_values[i]);
-
- max_filter_ = std::max(max_filter_, filter_length);
-}
-
-// BGRAConvolve2D -------------------------------------------------------------
-
-void BGRAConvolve2D(const uint8* source_data,
- int source_byte_row_stride,
- bool source_has_alpha,
- const ConvolusionFilter1D& filter_x,
- const ConvolusionFilter1D& filter_y,
- uint8* output) {
- int max_y_filter_size = filter_y.max_filter();
-
- // The next row in the input that we will generate a horizontally
- // convolved row for. If the filter doesn't start at the beginning of the
- // image (this is the case when we are only resizing a subset), then we
- // don't want to generate any output rows before that. Compute the starting
- // row for convolusion as the first pixel for the first vertical filter.
- int filter_offset, filter_length;
- const int16* filter_values =
- filter_y.FilterForValue(0, &filter_offset, &filter_length);
- int next_x_row = filter_offset;
-
- // We loop over each row in the input doing a horizontal convolusion. This
- // will result in a horizontally convolved image. We write the results into
- // a circular buffer of convolved rows and do vertical convolusion as rows
- // are available. This prevents us from having to store the entire
- // intermediate image and helps cache coherency.
- CircularRowBuffer row_buffer(filter_x.num_values(), max_y_filter_size,
- filter_offset);
-
- // Loop over every possible output row, processing just enough horizontal
- // convolusions to run each subsequent vertical convolusion.
- int output_row_byte_width = filter_x.num_values() * 4;
- int num_output_rows = filter_y.num_values();
- for (int out_y = 0; out_y < num_output_rows; out_y++) {
- filter_values = filter_y.FilterForValue(out_y,
- &filter_offset, &filter_length);
-
- // Generate output rows until we have enough to run the current filter.
- while (next_x_row < filter_offset + filter_length) {
- if (source_has_alpha) {
- ConvolveHorizontally<true>(
- &source_data[next_x_row * source_byte_row_stride],
- filter_x, row_buffer.AdvanceRow());
- } else {
- ConvolveHorizontally<false>(
- &source_data[next_x_row * source_byte_row_stride],
- filter_x, row_buffer.AdvanceRow());
- }
- next_x_row++;
- }
-
- // Compute where in the output image this row of final data will go.
- uint8* cur_output_row = &output[out_y * output_row_byte_width];
-
- // Get the list of rows that the circular buffer has, in order.
- int first_row_in_circular_buffer;
- uint8* const* rows_to_convolve =
- row_buffer.GetRowAddresses(&first_row_in_circular_buffer);
-
- // Now compute the start of the subset of those rows that the filter
- // needs.
- uint8* const* first_row_for_filter =
- &rows_to_convolve[filter_offset - first_row_in_circular_buffer];
-
- if (source_has_alpha) {
- ConvolveVertically<true>(filter_values, filter_length,
- first_row_for_filter,
- filter_x.num_values(), cur_output_row);
- } else {
- ConvolveVertically<false>(filter_values, filter_length,
- first_row_for_filter,
- filter_x.num_values(), cur_output_row);
- }
- }
-}
-
-} // namespace gfx
-
diff --git a/base/gfx/convolver.h b/base/gfx/convolver.h
deleted file mode 100644
index 12c9228..0000000
--- a/base/gfx/convolver.h
+++ /dev/null
@@ -1,137 +0,0 @@
-// Copyright (c) 2006-2008 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#ifndef BASE_GFX_CONVOLVER_H__
-#define BASE_GFX_CONVOLVER_H__
-
-#include <vector>
-
-#include "base/basictypes.h"
-
-// avoid confusion with Mac OS X's math library (Carbon)
-#if defined(OS_MACOSX)
-#undef FloatToFixed
-#endif
-
-namespace gfx {
-
-// Represents a filter in one dimension. Each output pixel has one entry in this
-// object for the filter values contributing to it. You build up the filter
-// list by calling AddFilter for each output pixel (in order).
-//
-// We do 2-dimensional convolusion by first convolving each row by one
-// ConvolusionFilter1D, then convolving each column by another one.
-//
-// Entries are stored in fixed point, shifted left by kShiftBits.
-class ConvolusionFilter1D {
- public:
- // The number of bits that fixed point values are shifted by.
- enum { kShiftBits = 14 };
-
- ConvolusionFilter1D() : max_filter_(0) {
- }
-
- // Convert between floating point and our fixed point representation.
- static inline int16 FloatToFixed(float f) {
- return static_cast<int16>(f * (1 << kShiftBits));
- }
- static inline unsigned char FixedToChar(int16 x) {
- return static_cast<unsigned char>(x >> kShiftBits);
- }
-
- // Returns the maximum pixel span of a filter.
- int max_filter() const { return max_filter_; }
-
- // Returns the number of filters in this filter. This is the dimension of the
- // output image.
- int num_values() const { return static_cast<int>(filters_.size()); }
-
- // Appends the given list of scaling values for generating a given output
- // pixel. |filter_offset| is the distance from the edge of the image to where
- // the scaling factors start. The scaling factors apply to the source pixels
- // starting from this position, and going for the next |filter_length| pixels.
- //
- // You will probably want to make sure your input is normalized (that is,
- // all entries in |filter_values| sub to one) to prevent affecting the overall
- // brighness of the image.
- //
- // The filter_length must be > 0.
- //
- // This version will automatically convert your input to fixed point.
- void AddFilter(int filter_offset,
- const float* filter_values,
- int filter_length);
-
- // Same as the above version, but the input is already fixed point.
- void AddFilter(int filter_offset,
- const int16* filter_values,
- int filter_length);
-
- // Retrieves a filter for the given |value_offset|, a position in the output
- // image in the direction we're convolving. The offset and length of the
- // filter values are put into the corresponding out arguments (see AddFilter
- // above for what these mean), and a pointer to the first scaling factor is
- // returned. There will be |filter_length| values in this array.
- inline const int16* FilterForValue(int value_offset,
- int* filter_offset,
- int* filter_length) const {
- const FilterInstance& filter = filters_[value_offset];
- *filter_offset = filter.offset;
- *filter_length = filter.length;
- return &filter_values_[filter.data_location];
- }
-
- private:
- struct FilterInstance {
- // Offset within filter_values for this instance of the filter.
- int data_location;
-
- // Distance from the left of the filter to the center. IN PIXELS
- int offset;
-
- // Number of values in this filter instance.
- int length;
- };
-
- // Stores the information for each filter added to this class.
- std::vector<FilterInstance> filters_;
-
- // We store all the filter values in this flat list, indexed by
- // |FilterInstance.data_location| to avoid the mallocs required for storing
- // each one separately.
- std::vector<int16> filter_values_;
-
- // The maximum size of any filter we've added.
- int max_filter_;
-};
-
-// Does a two-dimensional convolusion on the given source image.
-//
-// It is assumed the source pixel offsets referenced in the input filters
-// reference only valid pixels, so the source image size is not required. Each
-// row of the source image starts |source_byte_row_stride| after the previous
-// one (this allows you to have rows with some padding at the end).
-//
-// The result will be put into the given output buffer. The destination image
-// size will be xfilter.num_values() * yfilter.num_values() pixels. It will be
-// in rows of exactly xfilter.num_values() * 4 bytes.
-//
-// |source_has_alpha| is a hint that allows us to avoid doing computations on
-// the alpha channel if the image is opaque. If you don't know, set this to
-// true and it will work properly, but setting this to false will be a few
-// percent faster if you know the image is opaque.
-//
-// The layout in memory is assumed to be 4-bytes per pixel in B-G-R-A order
-// (this is ARGB when loaded into 32-bit words on a little-endian machine).
-void BGRAConvolve2D(const uint8* source_data,
- int source_byte_row_stride,
- bool source_has_alpha,
- const ConvolusionFilter1D& xfilter,
- const ConvolusionFilter1D& yfilter,
- uint8* output);
-
-} // namespace gfx
-
-#endif // BASE_GFX_CONVOLVER_H__
-
diff --git a/base/gfx/convolver_unittest.cc b/base/gfx/convolver_unittest.cc
deleted file mode 100644
index 0a30a6b..0000000
--- a/base/gfx/convolver_unittest.cc
+++ /dev/null
@@ -1,127 +0,0 @@
-// Copyright (c) 2006-2008 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#include <string.h>
-#include <time.h>
-#include <vector>
-
-#include "base/gfx/convolver.h"
-#include "testing/gtest/include/gtest/gtest.h"
-
-namespace gfx {
-
-namespace {
-
-// Fills the given filter with impulse functions for the range 0->num_entries.
- void FillImpulseFilter(int num_entries, ConvolusionFilter1D* filter) {
- float one = 1.0f;
- for (int i = 0; i < num_entries; i++)
- filter->AddFilter(i, &one, 1);
-}
-
-// Filters the given input with the impulse function, and verifies that it
-// does not change.
-void TestImpulseConvolusion(const unsigned char* data, int width, int height) {
- int byte_count = width * height * 4;
-
- ConvolusionFilter1D filter_x;
- FillImpulseFilter(width, &filter_x);
-
- ConvolusionFilter1D filter_y;
- FillImpulseFilter(height, &filter_y);
-
- std::vector<unsigned char> output;
- output.resize(byte_count);
- BGRAConvolve2D(data, width * 4, true, filter_x, filter_y, &output[0]);
-
- // Output should exactly match input.
- EXPECT_EQ(0, memcmp(data, &output[0], byte_count));
-}
-
-// Fills the destination filter with a box filter averaging every two pixels
-// to produce the output.
-void FillBoxFilter(int size, ConvolusionFilter1D* filter) {
- const float box[2] = { 0.5, 0.5 };
- for (int i = 0; i < size; i++)
- filter->AddFilter(i * 2, box, 2);
-}
-
-} // namespace
-
-// Tests that each pixel, when set and run through the impulse filter, does
-// not change.
-TEST(Convolver, Impulse) {
- // We pick an "odd" size that is not likely to fit on any boundaries so that
- // we can see if all the widths and paddings are handled properly.
- int width = 15;
- int height = 31;
- int byte_count = width * height * 4;
- std::vector<unsigned char> input;
- input.resize(byte_count);
-
- unsigned char* input_ptr = &input[0];
- for (int y = 0; y < height; y++) {
- for (int x = 0; x < width; x++) {
- for (int channel = 0; channel < 3; channel++) {
- memset(input_ptr, 0, byte_count);
- input_ptr[(y * width + x) * 4 + channel] = 0xff;
- // Always set the alpha channel or it will attempt to "fix" it for us.
- input_ptr[(y * width + x) * 4 + 3] = 0xff;
- TestImpulseConvolusion(input_ptr, width, height);
- }
- }
- }
-}
-
-// Tests that using a box filter to halve an image results in every square of 4
-// pixels in the original get averaged to a pixel in the output.
-TEST(Convolver, Halve) {
- static const int kSize = 16;
-
- int src_width = kSize;
- int src_height = kSize;
- int src_row_stride = src_width * 4;
- int src_byte_count = src_row_stride * src_height;
- std::vector<unsigned char> input;
- input.resize(src_byte_count);
-
- int dest_width = src_width / 2;
- int dest_height = src_height / 2;
- int dest_byte_count = dest_width * dest_height * 4;
- std::vector<unsigned char> output;
- output.resize(dest_byte_count);
-
- // First fill the array with a bunch of random data.
- srand(static_cast<unsigned>(time(NULL)));
- for (int i = 0; i < src_byte_count; i++)
- input[i] = rand() * 255 / RAND_MAX;
-
- // Compute the filters.
- ConvolusionFilter1D filter_x, filter_y;
- FillBoxFilter(dest_width, &filter_x);
- FillBoxFilter(dest_height, &filter_y);
-
- // Do the convolusion.
- BGRAConvolve2D(&input[0], src_width, true, filter_x, filter_y, &output[0]);
-
- // Compute the expected results and check, allowing for a small difference
- // to account for rounding errors.
- for (int y = 0; y < dest_height; y++) {
- for (int x = 0; x < dest_width; x++) {
- for (int channel = 0; channel < 4; channel++) {
- int src_offset = (y * 2 * src_row_stride + x * 2 * 4) + channel;
- int value = input[src_offset] + // Top left source pixel.
- input[src_offset + 4] + // Top right source pixel.
- input[src_offset + src_row_stride] + // Lower left.
- input[src_offset + src_row_stride + 4]; // Lower right.
- value /= 4; // Average.
- int difference = value - output[(y * dest_width + x) * 4 + channel];
- EXPECT_TRUE(difference >= -1 || difference <= 1);
- }
- }
- }
-}
-
-} // namespace gfx
-
diff --git a/base/gfx/image_operations.cc b/base/gfx/image_operations.cc
deleted file mode 100644
index a60d19e..0000000
--- a/base/gfx/image_operations.cc
+++ /dev/null
@@ -1,362 +0,0 @@
-// Copyright (c) 2006-2008 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-//
-#define _USE_MATH_DEFINES
-#include <cmath>
-#include <limits>
-#include <vector>
-
-#include "base/gfx/image_operations.h"
-
-#include "base/gfx/convolver.h"
-#include "base/gfx/rect.h"
-#include "base/gfx/size.h"
-#include "base/logging.h"
-#include "base/stack_container.h"
-#include "SkBitmap.h"
-
-namespace gfx {
-
-namespace {
-
-// Returns the ceiling/floor as an integer.
-inline int CeilInt(float val) {
- return static_cast<int>(ceil(val));
-}
-inline int FloorInt(float val) {
- return static_cast<int>(floor(val));
-}
-
-// Filter function computation -------------------------------------------------
-
-// Evaluates the box filter, which goes from -0.5 to +0.5.
-float EvalBox(float x) {
- return (x >= -0.5f && x < 0.5f) ? 1.0f : 0.0f;
-}
-
-// Evaluates the Lanczos filter of the given filter size window for the given
-// position.
-//
-// |filter_size| is the width of the filter (the "window"), outside of which
-// the value of the function is 0. Inside of the window, the value is the
-// normalized sinc function:
-// lanczos(x) = sinc(x) * sinc(x / filter_size);
-// where
-// sinc(x) = sin(pi*x) / (pi*x);
-float EvalLanczos(int filter_size, float x) {
- if (x <= -filter_size || x >= filter_size)
- return 0.0f; // Outside of the window.
- if (x > -std::numeric_limits<float>::epsilon() &&
- x < std::numeric_limits<float>::epsilon())
- return 1.0f; // Special case the discontinuity at the origin.
- float xpi = x * static_cast<float>(M_PI);
- return (sin(xpi) / xpi) * // sinc(x)
- sin(xpi / filter_size) / (xpi / filter_size); // sinc(x/filter_size)
-}
-
-// ResizeFilter ----------------------------------------------------------------
-
-// Encapsulates computation and storage of the filters required for one complete
-// resize operation.
-class ResizeFilter {
- public:
- ResizeFilter(ImageOperations::ResizeMethod method,
- const Size& src_full_size,
- const Size& dest_size,
- const Rect& dest_subset);
-
- // Returns the bounds in the input bitmap of data that is used in the output.
- // The filter offsets are within this rectangle.
- const Rect& src_depend() { return src_depend_; }
-
- // Returns the filled filter values.
- const ConvolusionFilter1D& x_filter() { return x_filter_; }
- const ConvolusionFilter1D& y_filter() { return y_filter_; }
-
- private:
- // Returns the number of pixels that the filer spans, in filter space (the
- // destination image).
- float GetFilterSupport(float scale) {
- switch (method_) {
- case ImageOperations::RESIZE_BOX:
- // The box filter just scales with the image scaling.
- return 0.5f; // Only want one side of the filter = /2.
- case ImageOperations::RESIZE_LANCZOS3:
- // The lanczos filter takes as much space in the source image in
- // each direction as the size of the window = 3 for Lanczos3.
- return 3.0f;
- default:
- NOTREACHED();
- return 1.0f;
- }
- }
-
- // Computes one set of filters either horizontally or vertically. The caller
- // will specify the "min" and "max" rather than the bottom/top and
- // right/bottom so that the same code can be re-used in each dimension.
- //
- // |src_depend_lo| and |src_depend_size| gives the range for the source
- // depend rectangle (horizontally or vertically at the caller's discretion
- // -- see above for what this means).
- //
- // Likewise, the range of destination values to compute and the scale factor
- // for the transform is also specified.
- void ComputeFilters(int src_size,
- int dest_subset_lo, int dest_subset_size,
- float scale, float src_support,
- ConvolusionFilter1D* output);
-
- // Computes the filter value given the coordinate in filter space.
- inline float ComputeFilter(float pos) {
- switch (method_) {
- case ImageOperations::RESIZE_BOX:
- return EvalBox(pos);
- case ImageOperations::RESIZE_LANCZOS3:
- return EvalLanczos(3, pos);
- default:
- NOTREACHED();
- return 0;
- }
- }
-
- ImageOperations::ResizeMethod method_;
-
- // Subset of source the filters will touch.
- Rect src_depend_;
-
- // Size of the filter support on one side only in the destination space.
- // See GetFilterSupport.
- float x_filter_support_;
- float y_filter_support_;
-
- // Subset of scaled destination bitmap to compute.
- Rect out_bounds_;
-
- ConvolusionFilter1D x_filter_;
- ConvolusionFilter1D y_filter_;
-
- DISALLOW_EVIL_CONSTRUCTORS(ResizeFilter);
-};
-
-ResizeFilter::ResizeFilter(ImageOperations::ResizeMethod method,
- const Size& src_full_size,
- const Size& dest_size,
- const Rect& dest_subset)
- : method_(method),
- out_bounds_(dest_subset) {
- float scale_x = static_cast<float>(dest_size.width()) /
- static_cast<float>(src_full_size.width());
- float scale_y = static_cast<float>(dest_size.height()) /
- static_cast<float>(src_full_size.height());
-
- x_filter_support_ = GetFilterSupport(scale_x);
- y_filter_support_ = GetFilterSupport(scale_y);
-
- gfx::Rect src_full(0, 0, src_full_size.width(), src_full_size.height());
- gfx::Rect dest_full(0, 0,
- static_cast<int>(src_full_size.width() * scale_x + 0.5),
- static_cast<int>(src_full_size.height() * scale_y + 0.5));
-
- // Support of the filter in source space.
- float src_x_support = x_filter_support_ / scale_x;
- float src_y_support = y_filter_support_ / scale_y;
-
- ComputeFilters(src_full_size.width(), dest_subset.x(), dest_subset.width(),
- scale_x, src_x_support, &x_filter_);
- ComputeFilters(src_full_size.height(), dest_subset.y(), dest_subset.height(),
- scale_y, src_y_support, &y_filter_);
-}
-
-void ResizeFilter::ComputeFilters(int src_size,
- int dest_subset_lo, int dest_subset_size,
- float scale, float src_support,
- ConvolusionFilter1D* output) {
- int dest_subset_hi = dest_subset_lo + dest_subset_size; // [lo, hi)
-
- // When we're doing a magnification, the scale will be larger than one. This
- // means the destination pixels are much smaller than the source pixels, and
- // that the range covered by the filter won't necessarily cover any source
- // pixel boundaries. Therefore, we use these clamped values (max of 1) for
- // some computations.
- float clamped_scale = std::min(1.0f, scale);
-
- // Speed up the divisions below by turning them into multiplies.
- float inv_scale = 1.0f / scale;
-
- StackVector<float, 64> filter_values;
- StackVector<int16, 64> fixed_filter_values;
-
- // Loop over all pixels in the output range. We will generate one set of
- // filter values for each one. Those values will tell us how to blend the
- // source pixels to compute the destination pixel.
- for (int dest_subset_i = dest_subset_lo; dest_subset_i < dest_subset_hi;
- dest_subset_i++) {
- // Reset the arrays. We don't declare them inside so they can re-use the
- // same malloc-ed buffer.
- filter_values->clear();
- fixed_filter_values->clear();
-
- // This is the pixel in the source directly under the pixel in the dest.
- float src_pixel = dest_subset_i * inv_scale;
-
- // Compute the (inclusive) range of source pixels the filter covers.
- int src_begin = std::max(0, FloorInt(src_pixel - src_support));
- int src_end = std::min(src_size - 1, CeilInt(src_pixel + src_support));
-
- // Compute the unnormalized filter value at each location of the source
- // it covers.
- float filter_sum = 0.0f; // Sub of the filter values for normalizing.
- for (int cur_filter_pixel = src_begin; cur_filter_pixel <= src_end;
- cur_filter_pixel++) {
- // Distance from the center of the filter, this is the filter coordinate
- // in source space.
- float src_filter_pos = cur_filter_pixel - src_pixel;
-
- // Since the filter really exists in dest space, map it there.
- float dest_filter_pos = src_filter_pos * clamped_scale;
-
- // Compute the filter value at that location.
- float filter_value = ComputeFilter(dest_filter_pos);
- filter_values->push_back(filter_value);
-
- filter_sum += filter_value;
- }
- DCHECK(!filter_values->empty()) << "We should always get a filter!";
-
- // The filter must be normalized so that we don't affect the brightness of
- // the image. Convert to normalized fixed point.
- int16 fixed_sum = 0;
- for (size_t i = 0; i < filter_values->size(); i++) {
- int16 cur_fixed = output->FloatToFixed(filter_values[i] / filter_sum);
- fixed_sum += cur_fixed;
- fixed_filter_values->push_back(cur_fixed);
- }
-
- // The conversion to fixed point will leave some rounding errors, which
- // we add back in to avoid affecting the brightness of the image. We
- // arbitrarily add this to the center of the filter array (this won't always
- // be the center of the filter function since it could get clipped on the
- // edges, but it doesn't matter enough to worry about that case).
- int16 leftovers = output->FloatToFixed(1.0f) - fixed_sum;
- fixed_filter_values[fixed_filter_values->size() / 2] += leftovers;
-
- // Now it's ready to go.
- output->AddFilter(src_begin, &fixed_filter_values[0],
- static_cast<int>(fixed_filter_values->size()));
- }
-}
-
-} // namespace
-
-// Resize ----------------------------------------------------------------------
-
-// static
-SkBitmap ImageOperations::Resize(const SkBitmap& source,
- ResizeMethod method,
- const Size& dest_size,
- const Rect& dest_subset) {
- DCHECK(Rect(dest_size.width(), dest_size.height()).Contains(dest_subset)) <<
- "The supplied subset does not fall within the destination image.";
-
- // If the size of source or destination is 0, i.e. 0x0, 0xN or Nx0, just
- // return empty
- if (source.width() < 1 || source.height() < 1 ||
- dest_size.width() < 1 || dest_size.height() < 1)
- return SkBitmap();
-
- SkAutoLockPixels locker(source);
-
- ResizeFilter filter(method, Size(source.width(), source.height()),
- dest_size, dest_subset);
-
- // Get a source bitmap encompassing this touched area. We construct the
- // offsets and row strides such that it looks like a new bitmap, while
- // referring to the old data.
- const uint8* source_subset =
- reinterpret_cast<const uint8*>(source.getPixels());
-
- // Convolve into the result.
- SkBitmap result;
- result.setConfig(SkBitmap::kARGB_8888_Config,
- dest_subset.width(), dest_subset.height());
- result.allocPixels();
- BGRAConvolve2D(source_subset, static_cast<int>(source.rowBytes()),
- !source.isOpaque(), filter.x_filter(), filter.y_filter(),
- static_cast<unsigned char*>(result.getPixels()));
-
- // Preserve the "opaque" flag for use as an optimization later.
- result.setIsOpaque(source.isOpaque());
-
- return result;
-}
-
-// static
-SkBitmap ImageOperations::Resize(const SkBitmap& source,
- ResizeMethod method,
- const Size& dest_size) {
- Rect dest_subset(0, 0, dest_size.width(), dest_size.height());
- return Resize(source, method, dest_size, dest_subset);
-}
-
-// static
-SkBitmap ImageOperations::CreateBlendedBitmap(const SkBitmap& first,
- const SkBitmap& second,
- double alpha) {
- DCHECK(alpha <= 1 && alpha >= 0);
- DCHECK(first.width() == second.width());
- DCHECK(first.height() == second.height());
- DCHECK(first.bytesPerPixel() == second.bytesPerPixel());
- DCHECK(first.config() == SkBitmap::kARGB_8888_Config);
-
- // Optimize for case where we won't need to blend anything.
- static const double alpha_min = 1.0 / 255;
- static const double alpha_max = 254.0 / 255;
- if (alpha < alpha_min) {
- return first;
- } else if (alpha > alpha_max) {
- return second;
- }
-
- SkAutoLockPixels lock_first(first);
- SkAutoLockPixels lock_second(second);
-
- SkBitmap blended;
- blended.setConfig(SkBitmap::kARGB_8888_Config, first.width(),
- first.height(), 0);
- blended.allocPixels();
- blended.eraseARGB(0, 0, 0, 0);
-
- double first_alpha = 1 - alpha;
-
- for (int y = 0; y < first.height(); y++) {
- uint32* first_row = first.getAddr32(0, y);
- uint32* second_row = second.getAddr32(0, y);
- uint32* dst_row = blended.getAddr32(0, y);
-
- for (int x = 0; x < first.width(); x++) {
- uint32 first_pixel = first_row[x];
- uint32 second_pixel = second_row[x];
-
- int a = static_cast<int>(
- SkColorGetA(first_pixel) * first_alpha +
- SkColorGetA(second_pixel) * alpha);
- int r = static_cast<int>(
- SkColorGetR(first_pixel) * first_alpha +
- SkColorGetR(second_pixel) * alpha);
- int g = static_cast<int>(
- SkColorGetG(first_pixel) * first_alpha +
- SkColorGetG(second_pixel) * alpha);
- int b = static_cast<int>(
- SkColorGetB(first_pixel) * first_alpha +
- SkColorGetB(second_pixel) * alpha);
-
- dst_row[x] = SkColorSetARGB(a, r, g, b);
- }
- }
-
- return blended;
-}
-
-} // namespace gfx
-
diff --git a/base/gfx/image_operations.h b/base/gfx/image_operations.h
deleted file mode 100644
index 826e651..0000000
--- a/base/gfx/image_operations.h
+++ /dev/null
@@ -1,63 +0,0 @@
-// Copyright (c) 2006-2008 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#ifndef BASE_GFX_IMAGE_OPERATIONS_H__
-#define BASE_GFX_IMAGE_OPERATIONS_H__
-
-#include "base/basictypes.h"
-#include "base/gfx/rect.h"
-
-class SkBitmap;
-
-namespace gfx {
-
-class ImageOperations {
- public:
- enum ResizeMethod {
- // Box filter. This is a weighted average of all of the pixels touching
- // the destination pixel. For enlargement, this is nearest neighbor.
- //
- // You probably don't want this, it is here for testing since it is easy to
- // compute. Use RESIZE_LANCZOS3 instead.
- RESIZE_BOX,
-
- // 3-cycle Lanczos filter. This is tall in the middle, goes negative on
- // each side, then oscillates 2 more times. It gives nice sharp edges.
- RESIZE_LANCZOS3,
- };
-
- // Resizes the given source bitmap using the specified resize method, so that
- // the entire image is (dest_size) big. The dest_subset is the rectangle in
- // this destination image that should actually be returned.
- //
- // The output image will be (dest_subset.width(), dest_subset.height()). This
- // will save work if you do not need the entire bitmap.
- //
- // The destination subset must be smaller than the destination image.
- static SkBitmap Resize(const SkBitmap& source,
- ResizeMethod method,
- const Size& dest_size,
- const Rect& dest_subset);
-
- // Alternate version for resizing and returning the entire bitmap rather than
- // a subset.
- static SkBitmap Resize(const SkBitmap& source,
- ResizeMethod method,
- const Size& dest_size);
-
-
- // Create a bitmap that is a blend of two others. The alpha argument
- // specifies the opacity of the second bitmap. The provided bitmaps must
- // use have the kARGB_8888_Config config and be of equal dimensions.
- static SkBitmap CreateBlendedBitmap(const SkBitmap& first,
- const SkBitmap& second,
- double alpha);
- private:
- ImageOperations(); // Class for scoping only.
-};
-
-} // namespace gfx
-
-#endif // BASE_GFX_IMAGE_OPERATIONS_H__
-
diff --git a/base/gfx/image_operations_unittest.cc b/base/gfx/image_operations_unittest.cc
deleted file mode 100644
index a15e648..0000000
--- a/base/gfx/image_operations_unittest.cc
+++ /dev/null
@@ -1,148 +0,0 @@
-// Copyright (c) 2006-2008 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#include <stdlib.h>
-
-#include "base/gfx/image_operations.h"
-#include "testing/gtest/include/gtest/gtest.h"
-#include "SkBitmap.h"
-
-namespace {
-
-// Computes the average pixel value for the given range, inclusive.
-uint32_t AveragePixel(const SkBitmap& bmp,
- int x_min, int x_max,
- int y_min, int y_max) {
- float accum[4] = {0, 0, 0, 0};
- int count = 0;
- for (int y = y_min; y <= y_max; y++) {
- for (int x = x_min; x <= x_max; x++) {
- uint32_t cur = *bmp.getAddr32(x, y);
- accum[0] += SkColorGetB(cur);
- accum[1] += SkColorGetG(cur);
- accum[2] += SkColorGetR(cur);
- accum[3] += SkColorGetA(cur);
- count++;
- }
- }
-
- return SkColorSetARGB(static_cast<unsigned char>(accum[3] / count),
- static_cast<unsigned char>(accum[2] / count),
- static_cast<unsigned char>(accum[1] / count),
- static_cast<unsigned char>(accum[0] / count));
-}
-
-// Returns true if each channel of the given two colors are "close." This is
-// used for comparing colors where rounding errors may cause off-by-one.
-bool ColorsClose(uint32_t a, uint32_t b) {
- return abs(static_cast<int>(SkColorGetB(a) - SkColorGetB(b))) < 2 &&
- abs(static_cast<int>(SkColorGetG(a) - SkColorGetG(b))) < 2 &&
- abs(static_cast<int>(SkColorGetR(a) - SkColorGetR(b))) < 2 &&
- abs(static_cast<int>(SkColorGetA(a) - SkColorGetA(b))) < 2;
-}
-
-void FillDataToBitmap(int w, int h, SkBitmap* bmp) {
- bmp->setConfig(SkBitmap::kARGB_8888_Config, w, h);
- bmp->allocPixels();
-
- unsigned char* src_data =
- reinterpret_cast<unsigned char*>(bmp->getAddr32(0, 0));
- for (int i = 0; i < w * h; i++) {
- src_data[i * 4 + 0] = static_cast<unsigned char>(i % 255);
- src_data[i * 4 + 1] = static_cast<unsigned char>(i % 255);
- src_data[i * 4 + 2] = static_cast<unsigned char>(i % 255);
- src_data[i * 4 + 3] = static_cast<unsigned char>(i % 255);
- }
-}
-
-} // namespace
-
-// Makes the bitmap 50% the size as the original using a box filter. This is
-// an easy operation that we can check the results for manually.
-TEST(ImageOperations, Halve) {
- // Make our source bitmap.
- int src_w = 30, src_h = 38;
- SkBitmap src;
- FillDataToBitmap(src_w, src_h, &src);
-
- // Do a halving of the full bitmap.
- SkBitmap actual_results = gfx::ImageOperations::Resize(
- src, gfx::ImageOperations::RESIZE_BOX, gfx::Size(src_w / 2, src_h / 2));
- ASSERT_EQ(src_w / 2, actual_results.width());
- ASSERT_EQ(src_h / 2, actual_results.height());
-
- // Compute the expected values & compare.
- SkAutoLockPixels lock(actual_results);
- for (int y = 0; y < actual_results.height(); y++) {
- for (int x = 0; x < actual_results.width(); x++) {
- int first_x = std::max(0, x * 2 - 1);
- int last_x = std::min(src_w - 1, x * 2);
-
- int first_y = std::max(0, y * 2 - 1);
- int last_y = std::min(src_h - 1, y * 2);
-
- uint32_t expected_color = AveragePixel(src,
- first_x, last_x, first_y, last_y);
- EXPECT_TRUE(ColorsClose(expected_color, *actual_results.getAddr32(x, y)));
- }
- }
-}
-
-TEST(ImageOperations, HalveSubset) {
- // Make our source bitmap.
- int src_w = 16, src_h = 34;
- SkBitmap src;
- FillDataToBitmap(src_w, src_h, &src);
-
- // Do a halving of the full bitmap.
- SkBitmap full_results = gfx::ImageOperations::Resize(
- src, gfx::ImageOperations::RESIZE_BOX, gfx::Size(src_w / 2, src_h / 2));
- ASSERT_EQ(src_w / 2, full_results.width());
- ASSERT_EQ(src_h / 2, full_results.height());
-
- // Now do a halving of a a subset, recall the destination subset is in the
- // destination coordinate system (max = half of the original image size).
- gfx::Rect subset_rect(2, 3, 3, 6);
- SkBitmap subset_results = gfx::ImageOperations::Resize(
- src, gfx::ImageOperations::RESIZE_BOX,
- gfx::Size(src_w / 2, src_h / 2), subset_rect);
- ASSERT_EQ(subset_rect.width(), subset_results.width());
- ASSERT_EQ(subset_rect.height(), subset_results.height());
-
- // The computed subset and the corresponding subset of the original image
- // should be the same.
- SkAutoLockPixels full_lock(full_results);
- SkAutoLockPixels subset_lock(subset_results);
- for (int y = 0; y < subset_rect.height(); y++) {
- for (int x = 0; x < subset_rect.width(); x++) {
- ASSERT_EQ(
- *full_results.getAddr32(x + subset_rect.x(), y + subset_rect.y()),
- *subset_results.getAddr32(x, y));
- }
- }
-}
-
-// Resamples an iamge to the same image, it should give almost the same result.
-TEST(ImageOperations, ResampleToSame) {
- // Make our source bitmap.
- int src_w = 16, src_h = 34;
- SkBitmap src;
- FillDataToBitmap(src_w, src_h, &src);
-
- // Do a resize of the full bitmap to the same size. The lanczos filter is good
- // enough that we should get exactly the same image for output.
- SkBitmap results = gfx::ImageOperations::Resize(
- src, gfx::ImageOperations::RESIZE_LANCZOS3, gfx::Size(src_w, src_h));
- ASSERT_EQ(src_w, results.width());
- ASSERT_EQ(src_h, results.height());
-
- SkAutoLockPixels src_lock(src);
- SkAutoLockPixels results_lock(results);
- for (int y = 0; y < src_h; y++) {
- for (int x = 0; x < src_w; x++) {
- EXPECT_EQ(*src.getAddr32(x, y), *results.getAddr32(x, y));
- }
- }
-}
-
diff --git a/base/gfx/img_resize_perftest.cc b/base/gfx/img_resize_perftest.cc
deleted file mode 100644
index 6a4b070..0000000
--- a/base/gfx/img_resize_perftest.cc
+++ /dev/null
@@ -1,70 +0,0 @@
-// Copyright (c) 2006-2008 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#include <stdlib.h>
-#include <time.h>
-
-#include "base/perftimer.h"
-#include "base/gfx/convolver.h"
-#include "base/gfx/image_operations.h"
-#include "base/gfx/image_resizer.h"
-#include "testing/gtest/include/gtest/gtest.h"
-
-namespace {
-
-void FillRandomData(char* dest, int byte_count) {
- srand(static_cast<unsigned>(time(NULL)));
- for (int i = 0; i < byte_count; i++)
- dest[i] = rand() * 255 / RAND_MAX;
-}
-
-} // namespace
-
-// Old code gives [1521, 1519]ms for this, 4000x4000 -> 2100x2100 lanczos8
-
-TEST(ImageResizePerf, BigFilter) {
- static const int kSrcWidth = 4000;
- static const int kSrcHeight = 4000;
- static const int kSrcByteSize = kSrcWidth * kSrcHeight * 4;
-
- SkBitmap src_bmp;
- src_bmp.setConfig(SkBitmap::kARGB_8888_Config, kSrcWidth, kSrcHeight);
- src_bmp.allocPixels();
- FillRandomData(reinterpret_cast<char*>(src_bmp.getAddr32(0, 0)),
- kSrcByteSize);
-
- // Make the dest size > 1/2 so the 50% optimization doesn't kick in.
- static const int kDestWidth = 1400;
- static const int kDestHeight = 1400;
-
- PerfTimeLogger resize_timer("resize");
- gfx::ImageResizer resizer(gfx::ImageResizer::LANCZOS3);
- SkBitmap dest = resizer.Resize(src_bmp, kDestWidth, kDestHeight);
-}
-
-// The original image filter we were using took 523ms for this test, while this
-// one takes 857ms.
-// TODO(brettw) make this at least 64% faster.
-TEST(ImageOperationPerf, BigFilter) {
- static const int kSrcWidth = 4000;
- static const int kSrcHeight = 4000;
- static const int kSrcByteSize = kSrcWidth * kSrcHeight * 4;
-
- SkBitmap src_bmp;
- src_bmp.setConfig(SkBitmap::kARGB_8888_Config, kSrcWidth, kSrcHeight);
- src_bmp.allocPixels();
- src_bmp.setIsOpaque(true);
- FillRandomData(reinterpret_cast<char*>(src_bmp.getAddr32(0, 0)),
- kSrcByteSize);
-
- // Make the dest size > 1/2 so the 50% optimization doesn't kick in.
- static const int kDestWidth = 1400;
- static const int kDestHeight = 1400;
-
- PerfTimeLogger resize_timer("resize");
- SkBitmap dest = gfx::ImageOperations::Resize(src_bmp,
- gfx::ImageOperations::RESIZE_LANCZOS3, (float)kDestWidth / (float)kSrcWidth,
- (float)kDestHeight / (float)kSrcHeight);
-}
-