blob: 3c513b8c6daf9f8a0498d9e9ca59d9b8f40d94a0 [file] [log] [blame]
// Copyright 2017 The PDFium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
// Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com
#include "core/fxge/dib/cfx_imagetransformer.h"
#include <math.h>
#include <array>
#include <iterator>
#include <memory>
#include <utility>
#include "core/fxcrt/check.h"
#include "core/fxcrt/compiler_specific.h"
#include "core/fxcrt/fx_system.h"
#include "core/fxcrt/notreached.h"
#include "core/fxcrt/numerics/safe_conversions.h"
#include "core/fxcrt/stl_util.h"
#include "core/fxge/dib/cfx_dibitmap.h"
#include "core/fxge/dib/cfx_imagestretcher.h"
#include "core/fxge/dib/fx_dib.h"
namespace {
constexpr int kBase = 256;
constexpr float kFix16 = 0.05f;
constexpr uint8_t kOpaqueAlpha = 0xff;
uint8_t BilinearInterpolate(const uint8_t* buf,
const CFX_ImageTransformer::BilinearData& data,
int bytes_per_pixel,
int c_offset) {
const int i_resx = 255 - data.res_x;
const int col_bpp_l = data.src_col_l * bytes_per_pixel;
const int col_bpp_r = data.src_col_r * bytes_per_pixel;
UNSAFE_TODO({
const uint8_t* buf_u = buf + data.row_offset_l + c_offset;
const uint8_t* buf_d = buf + data.row_offset_r + c_offset;
const uint8_t* src_pos0 = buf_u + col_bpp_l;
const uint8_t* src_pos1 = buf_u + col_bpp_r;
const uint8_t* src_pos2 = buf_d + col_bpp_l;
const uint8_t* src_pos3 = buf_d + col_bpp_r;
uint8_t r_pos_0 = (*src_pos0 * i_resx + *src_pos1 * data.res_x) >> 8;
uint8_t r_pos_1 = (*src_pos2 * i_resx + *src_pos3 * data.res_x) >> 8;
return (r_pos_0 * (255 - data.res_y) + r_pos_1 * data.res_y) >> 8;
});
}
class CFX_BilinearMatrix {
public:
explicit CFX_BilinearMatrix(const CFX_Matrix& src)
: a(FXSYS_roundf(src.a * kBase)),
b(FXSYS_roundf(src.b * kBase)),
c(FXSYS_roundf(src.c * kBase)),
d(FXSYS_roundf(src.d * kBase)),
e(FXSYS_roundf(src.e * kBase)),
f(FXSYS_roundf(src.f * kBase)) {}
void Transform(int x, int y, int* x1, int* y1, int* res_x, int* res_y) const {
CFX_PointF val = TransformInternal(CFX_PointF(x, y));
*x1 = pdfium::saturated_cast<int>(val.x / kBase);
*y1 = pdfium::saturated_cast<int>(val.y / kBase);
*res_x = static_cast<int>(val.x) % kBase;
*res_y = static_cast<int>(val.y) % kBase;
if (*res_x < 0 && *res_x > -kBase) {
*res_x = kBase + *res_x;
}
if (*res_y < 0 && *res_y > -kBase) {
*res_y = kBase + *res_y;
}
}
private:
CFX_PointF TransformInternal(CFX_PointF pt) const {
return CFX_PointF(a * pt.x + c * pt.y + e + kBase / 2,
b * pt.x + d * pt.y + f + kBase / 2);
}
const int a;
const int b;
const int c;
const int d;
const int e;
const int f;
};
bool InStretchBounds(const FX_RECT& clip_rect, int col, int row) {
return col >= 0 && col <= clip_rect.Width() && row >= 0 &&
row <= clip_rect.Height();
}
void AdjustCoords(const FX_RECT& clip_rect, int* col, int* row) {
int& src_col = *col;
int& src_row = *row;
if (src_col == clip_rect.Width()) {
src_col--;
}
if (src_row == clip_rect.Height()) {
src_row--;
}
}
// Let the compiler deduce the type for |func|, which cheaper than specifying it
// with std::function.
template <typename F>
void DoBilinearLoop(const CFX_ImageTransformer::CalcData& calc_data,
const FX_RECT& result_rect,
const FX_RECT& clip_rect,
int increment,
const F& func) {
CFX_BilinearMatrix matrix_fix(calc_data.matrix);
for (int row = 0; row < result_rect.Height(); row++) {
uint8_t* dest = calc_data.bitmap->GetWritableScanline(row).data();
for (int col = 0; col < result_rect.Width(); col++) {
CFX_ImageTransformer::BilinearData d;
d.res_x = 0;
d.res_y = 0;
d.src_col_l = 0;
d.src_row_l = 0;
matrix_fix.Transform(col, row, &d.src_col_l, &d.src_row_l, &d.res_x,
&d.res_y);
if (LIKELY(InStretchBounds(clip_rect, d.src_col_l, d.src_row_l))) {
AdjustCoords(clip_rect, &d.src_col_l, &d.src_row_l);
d.src_col_r = d.src_col_l + 1;
d.src_row_r = d.src_row_l + 1;
AdjustCoords(clip_rect, &d.src_col_r, &d.src_row_r);
d.row_offset_l = d.src_row_l * calc_data.pitch;
d.row_offset_r = d.src_row_r * calc_data.pitch;
func(d, dest);
}
UNSAFE_TODO(dest += increment);
}
}
}
} // namespace
CFX_ImageTransformer::CFX_ImageTransformer(RetainPtr<const CFX_DIBBase> source,
const CFX_Matrix& matrix,
const FXDIB_ResampleOptions& options,
const FX_RECT* pClip)
: src_(std::move(source)), matrix_(matrix), resample_options_(options) {
FX_RECT result_rect = matrix_.GetUnitRect().GetClosestRect();
FX_RECT result_clip = result_rect;
if (pClip) {
result_clip.Intersect(*pClip);
}
if (result_clip.IsEmpty()) {
return;
}
result_ = result_clip;
if (fabs(matrix_.a) < fabs(matrix_.b) / 20 &&
fabs(matrix_.d) < fabs(matrix_.c) / 20 && fabs(matrix_.a) < 0.5f &&
fabs(matrix_.d) < 0.5f) {
int dest_width = result_rect.Width();
int dest_height = result_rect.Height();
result_clip.Offset(-result_rect.left, -result_rect.top);
result_clip = result_clip.SwappedClipBox(dest_width, dest_height,
matrix_.c > 0, matrix_.b < 0);
stretcher_ = std::make_unique<CFX_ImageStretcher>(
&storer_, src_, dest_height, dest_width, result_clip,
resample_options_);
stretcher_->Start();
type_ = StretchType::kRotate;
return;
}
if (fabs(matrix_.b) < kFix16 && fabs(matrix_.c) < kFix16) {
int dest_width =
static_cast<int>(matrix_.a > 0 ? ceil(matrix_.a) : floor(matrix_.a));
int dest_height =
static_cast<int>(matrix_.d > 0 ? -ceil(matrix_.d) : -floor(matrix_.d));
result_clip.Offset(-result_rect.left, -result_rect.top);
stretcher_ = std::make_unique<CFX_ImageStretcher>(
&storer_, src_, dest_width, dest_height, result_clip,
resample_options_);
stretcher_->Start();
type_ = StretchType::kNormal;
return;
}
int stretch_width = static_cast<int>(ceil(hypotf(matrix_.a, matrix_.b)));
if (stretch_width == 0) {
return;
}
int stretch_height = static_cast<int>(ceil(hypotf(matrix_.c, matrix_.d)));
if (stretch_height == 0) {
return;
}
CFX_Matrix stretch_to_dest(1.0f, 0.0f, 0.0f, -1.0f, 0.0f, stretch_height);
stretch_to_dest.Concat(
CFX_Matrix(matrix_.a / stretch_width, matrix_.b / stretch_width,
matrix_.c / stretch_height, matrix_.d / stretch_height,
matrix_.e, matrix_.f));
CFX_Matrix dest_to_strech = stretch_to_dest.GetInverse();
FX_RECT stretch_clip =
dest_to_strech.TransformRect(CFX_FloatRect(result_clip)).GetOuterRect();
if (!stretch_clip.Valid()) {
return;
}
stretch_clip.Intersect(0, 0, stretch_width, stretch_height);
if (!stretch_clip.Valid()) {
return;
}
dest_to_stretch_ = dest_to_strech;
stretch_clip_ = stretch_clip;
stretcher_ = std::make_unique<CFX_ImageStretcher>(
&storer_, src_, stretch_width, stretch_height, stretch_clip_,
resample_options_);
stretcher_->Start();
type_ = StretchType::kOther;
}
CFX_ImageTransformer::~CFX_ImageTransformer() = default;
bool CFX_ImageTransformer::Continue(PauseIndicatorIface* pPause) {
if (type_ == StretchType::kNone) {
return false;
}
if (stretcher_->Continue(pPause)) {
return true;
}
switch (type_) {
case StretchType::kNone:
// Already handled separately at the beginning of this method.
NOTREACHED();
case StretchType::kNormal:
return false;
case StretchType::kRotate:
ContinueRotate(pPause);
return false;
case StretchType::kOther:
ContinueOther(pPause);
return false;
}
}
void CFX_ImageTransformer::ContinueRotate(PauseIndicatorIface* pPause) {
if (storer_.GetBitmap()) {
storer_.Replace(storer_.GetBitmap()->SwapXY(matrix_.c > 0, matrix_.b < 0));
}
}
void CFX_ImageTransformer::ContinueOther(PauseIndicatorIface* pPause) {
if (!storer_.GetBitmap()) {
return;
}
auto pTransformed = pdfium::MakeRetain<CFX_DIBitmap>();
// TODO(crbug.com/42271020): Consider adding support for
// `FXDIB_Format::kBgraPremul`
FXDIB_Format dest_format = stretcher_->source()->IsMaskFormat()
? FXDIB_Format::k8bppMask
: FXDIB_Format::kBgra;
if (!pTransformed->Create(result_.Width(), result_.Height(), dest_format)) {
return;
}
CFX_Matrix result2stretch(1.0f, 0.0f, 0.0f, 1.0f, result_.left, result_.top);
result2stretch.Concat(dest_to_stretch_);
result2stretch.Translate(-stretch_clip_.left, -stretch_clip_.top);
CalcData calc_data = {pTransformed.Get(), result2stretch,
storer_.GetBitmap()->GetBuffer().data(),
storer_.GetBitmap()->GetPitch()};
if (storer_.GetBitmap()->IsMaskFormat()) {
CalcAlpha(calc_data);
} else {
const int src_bytes_per_pixel = storer_.GetBitmap()->GetBPP() / 8;
if (src_bytes_per_pixel == 1) {
CalcMono(calc_data);
} else {
CalcColor(calc_data, dest_format, src_bytes_per_pixel);
}
}
storer_.Replace(std::move(pTransformed));
}
RetainPtr<CFX_DIBitmap> CFX_ImageTransformer::DetachBitmap() {
return storer_.Detach();
}
void CFX_ImageTransformer::CalcAlpha(const CalcData& calc_data) {
auto func = [&calc_data](const BilinearData& data, uint8_t* dest) {
*dest = BilinearInterpolate(calc_data.buf, data, 1, 0);
};
DoBilinearLoop(calc_data, result_, stretch_clip_, 1, func);
}
void CFX_ImageTransformer::CalcMono(const CalcData& calc_data) {
std::array<uint32_t, 256> argb;
if (storer_.GetBitmap()->HasPalette()) {
pdfium::span<const uint32_t> palette =
storer_.GetBitmap()->GetPaletteSpan();
fxcrt::Copy(palette.first(argb.size()), argb);
} else {
for (uint32_t i = 0; i < argb.size(); ++i) {
argb[i] = ArgbEncode(0xff, i, i, i);
}
}
const int dest_bytes_per_pixel = calc_data.bitmap->GetBPP() / 8;
auto func = [&calc_data, &argb](const BilinearData& data, uint8_t* dest) {
uint8_t idx = BilinearInterpolate(calc_data.buf, data, 1, 0);
*reinterpret_cast<uint32_t*>(dest) = argb[idx];
};
DoBilinearLoop(calc_data, result_, stretch_clip_, dest_bytes_per_pixel, func);
}
void CFX_ImageTransformer::CalcColor(const CalcData& calc_data,
FXDIB_Format dest_format,
int src_bytes_per_pixel) {
DCHECK(dest_format == FXDIB_Format::k8bppMask ||
dest_format == FXDIB_Format::kBgra);
const int dest_bytes_per_pixel = calc_data.bitmap->GetBPP() / 8;
if (!storer_.GetBitmap()->IsAlphaFormat()) {
auto func = [&calc_data, src_bytes_per_pixel](const BilinearData& data,
uint8_t* dest) {
uint8_t b =
BilinearInterpolate(calc_data.buf, data, src_bytes_per_pixel, 0);
uint8_t g =
BilinearInterpolate(calc_data.buf, data, src_bytes_per_pixel, 1);
uint8_t r =
BilinearInterpolate(calc_data.buf, data, src_bytes_per_pixel, 2);
*reinterpret_cast<uint32_t*>(dest) = ArgbEncode(kOpaqueAlpha, r, g, b);
};
DoBilinearLoop(calc_data, result_, stretch_clip_, dest_bytes_per_pixel,
func);
return;
}
if (dest_format == FXDIB_Format::kBgra) {
auto func = [&calc_data, src_bytes_per_pixel](const BilinearData& data,
uint8_t* dest) {
uint8_t b =
BilinearInterpolate(calc_data.buf, data, src_bytes_per_pixel, 0);
uint8_t g =
BilinearInterpolate(calc_data.buf, data, src_bytes_per_pixel, 1);
uint8_t r =
BilinearInterpolate(calc_data.buf, data, src_bytes_per_pixel, 2);
uint8_t alpha =
BilinearInterpolate(calc_data.buf, data, src_bytes_per_pixel, 3);
*reinterpret_cast<uint32_t*>(dest) = ArgbEncode(alpha, r, g, b);
};
DoBilinearLoop(calc_data, result_, stretch_clip_, dest_bytes_per_pixel,
func);
return;
}
auto func = [&calc_data, src_bytes_per_pixel](const BilinearData& data,
uint8_t* dest) {
uint8_t c =
BilinearInterpolate(calc_data.buf, data, src_bytes_per_pixel, 0);
uint8_t m =
BilinearInterpolate(calc_data.buf, data, src_bytes_per_pixel, 1);
uint8_t y =
BilinearInterpolate(calc_data.buf, data, src_bytes_per_pixel, 2);
uint8_t k =
BilinearInterpolate(calc_data.buf, data, src_bytes_per_pixel, 3);
*reinterpret_cast<uint32_t*>(dest) = FXCMYK_TODIB(CmykEncode(c, m, y, k));
};
DoBilinearLoop(calc_data, result_, stretch_clip_, dest_bytes_per_pixel, func);
}