401 lines
11 KiB
C
401 lines
11 KiB
C
/*
|
|
* Copyright 2010 Vincent Povirk for CodeWeavers
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <stdarg.h>
|
|
|
|
#define COBJMACROS
|
|
|
|
#include "windef.h"
|
|
#include "winbase.h"
|
|
#include "objbase.h"
|
|
#include "wincodec.h"
|
|
|
|
#include "wincodecs_private.h"
|
|
|
|
#include "wine/debug.h"
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(wincodecs);
|
|
|
|
typedef struct BitmapScaler {
|
|
IWICBitmapScaler IWICBitmapScaler_iface;
|
|
LONG ref;
|
|
IWICBitmapSource *source;
|
|
UINT width, height;
|
|
UINT src_width, src_height;
|
|
WICBitmapInterpolationMode mode;
|
|
UINT bpp;
|
|
void (*fn_get_required_source_rect)(struct BitmapScaler*,UINT,UINT,WICRect*);
|
|
void (*fn_copy_scanline)(struct BitmapScaler*,UINT,UINT,UINT,BYTE**,UINT,UINT,BYTE*);
|
|
CRITICAL_SECTION lock; /* must be held when initialized */
|
|
} BitmapScaler;
|
|
|
|
static inline BitmapScaler *impl_from_IWICBitmapScaler(IWICBitmapScaler *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, BitmapScaler, IWICBitmapScaler_iface);
|
|
}
|
|
|
|
static HRESULT WINAPI BitmapScaler_QueryInterface(IWICBitmapScaler *iface, REFIID iid,
|
|
void **ppv)
|
|
{
|
|
BitmapScaler *This = impl_from_IWICBitmapScaler(iface);
|
|
TRACE("(%p,%s,%p)\n", iface, debugstr_guid(iid), ppv);
|
|
|
|
if (!ppv) return E_INVALIDARG;
|
|
|
|
if (IsEqualIID(&IID_IUnknown, iid) ||
|
|
IsEqualIID(&IID_IWICBitmapSource, iid) ||
|
|
IsEqualIID(&IID_IWICBitmapScaler, iid))
|
|
{
|
|
*ppv = &This->IWICBitmapScaler_iface;
|
|
}
|
|
else
|
|
{
|
|
*ppv = NULL;
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
IUnknown_AddRef((IUnknown*)*ppv);
|
|
return S_OK;
|
|
}
|
|
|
|
static ULONG WINAPI BitmapScaler_AddRef(IWICBitmapScaler *iface)
|
|
{
|
|
BitmapScaler *This = impl_from_IWICBitmapScaler(iface);
|
|
ULONG ref = InterlockedIncrement(&This->ref);
|
|
|
|
TRACE("(%p) refcount=%u\n", iface, ref);
|
|
|
|
return ref;
|
|
}
|
|
|
|
static ULONG WINAPI BitmapScaler_Release(IWICBitmapScaler *iface)
|
|
{
|
|
BitmapScaler *This = impl_from_IWICBitmapScaler(iface);
|
|
ULONG ref = InterlockedDecrement(&This->ref);
|
|
|
|
TRACE("(%p) refcount=%u\n", iface, ref);
|
|
|
|
if (ref == 0)
|
|
{
|
|
This->lock.DebugInfo->Spare[0] = 0;
|
|
DeleteCriticalSection(&This->lock);
|
|
if (This->source) IWICBitmapSource_Release(This->source);
|
|
HeapFree(GetProcessHeap(), 0, This);
|
|
}
|
|
|
|
return ref;
|
|
}
|
|
|
|
static HRESULT WINAPI BitmapScaler_GetSize(IWICBitmapScaler *iface,
|
|
UINT *puiWidth, UINT *puiHeight)
|
|
{
|
|
BitmapScaler *This = impl_from_IWICBitmapScaler(iface);
|
|
TRACE("(%p,%p,%p)\n", iface, puiWidth, puiHeight);
|
|
|
|
if (!puiWidth || !puiHeight)
|
|
return E_INVALIDARG;
|
|
|
|
if (!This->source)
|
|
return WINCODEC_ERR_WRONGSTATE;
|
|
|
|
*puiWidth = This->width;
|
|
*puiHeight = This->height;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI BitmapScaler_GetPixelFormat(IWICBitmapScaler *iface,
|
|
WICPixelFormatGUID *pPixelFormat)
|
|
{
|
|
BitmapScaler *This = impl_from_IWICBitmapScaler(iface);
|
|
TRACE("(%p,%p)\n", iface, pPixelFormat);
|
|
|
|
if (!pPixelFormat)
|
|
return E_INVALIDARG;
|
|
|
|
if (!This->source)
|
|
return WINCODEC_ERR_WRONGSTATE;
|
|
|
|
return IWICBitmapSource_GetPixelFormat(This->source, pPixelFormat);
|
|
}
|
|
|
|
static HRESULT WINAPI BitmapScaler_GetResolution(IWICBitmapScaler *iface,
|
|
double *pDpiX, double *pDpiY)
|
|
{
|
|
BitmapScaler *This = impl_from_IWICBitmapScaler(iface);
|
|
TRACE("(%p,%p,%p)\n", iface, pDpiX, pDpiY);
|
|
|
|
if (!pDpiX || !pDpiY)
|
|
return E_INVALIDARG;
|
|
|
|
if (!This->source)
|
|
return WINCODEC_ERR_WRONGSTATE;
|
|
|
|
return IWICBitmapSource_GetResolution(This->source, pDpiX, pDpiY);
|
|
}
|
|
|
|
static HRESULT WINAPI BitmapScaler_CopyPalette(IWICBitmapScaler *iface,
|
|
IWICPalette *pIPalette)
|
|
{
|
|
BitmapScaler *This = impl_from_IWICBitmapScaler(iface);
|
|
TRACE("(%p,%p)\n", iface, pIPalette);
|
|
|
|
if (!pIPalette)
|
|
return E_INVALIDARG;
|
|
|
|
if (!This->source)
|
|
return WINCODEC_ERR_WRONGSTATE;
|
|
|
|
return IWICBitmapSource_CopyPalette(This->source, pIPalette);
|
|
}
|
|
|
|
static void NearestNeighbor_GetRequiredSourceRect(BitmapScaler *This,
|
|
UINT x, UINT y, WICRect *src_rect)
|
|
{
|
|
src_rect->X = x * This->src_width / This->width;
|
|
src_rect->Y = y * This->src_height / This->height;
|
|
src_rect->Width = src_rect->Height = 1;
|
|
}
|
|
|
|
static void NearestNeighbor_CopyScanline(BitmapScaler *This,
|
|
UINT dst_x, UINT dst_y, UINT dst_width,
|
|
BYTE **src_data, UINT src_data_x, UINT src_data_y, BYTE *pbBuffer)
|
|
{
|
|
int i;
|
|
UINT bytesperpixel = This->bpp/8;
|
|
UINT src_x, src_y;
|
|
|
|
src_y = dst_y * This->src_height / This->height - src_data_y;
|
|
|
|
for (i=0; i<dst_width; i++)
|
|
{
|
|
src_x = (dst_x + i) * This->src_width / This->width - src_data_x;
|
|
memcpy(pbBuffer + bytesperpixel * i, src_data[src_y] + bytesperpixel * src_x, bytesperpixel);
|
|
}
|
|
}
|
|
|
|
static HRESULT WINAPI BitmapScaler_CopyPixels(IWICBitmapScaler *iface,
|
|
const WICRect *prc, UINT cbStride, UINT cbBufferSize, BYTE *pbBuffer)
|
|
{
|
|
BitmapScaler *This = impl_from_IWICBitmapScaler(iface);
|
|
HRESULT hr;
|
|
WICRect dest_rect;
|
|
WICRect src_rect_ul, src_rect_br, src_rect;
|
|
BYTE **src_rows;
|
|
BYTE *src_bits;
|
|
ULONG bytesperrow;
|
|
ULONG src_bytesperrow;
|
|
ULONG buffer_size;
|
|
UINT y;
|
|
|
|
TRACE("(%p,%p,%u,%u,%p)\n", iface, prc, cbStride, cbBufferSize, pbBuffer);
|
|
|
|
EnterCriticalSection(&This->lock);
|
|
|
|
if (!This->source)
|
|
{
|
|
hr = WINCODEC_ERR_WRONGSTATE;
|
|
goto end;
|
|
}
|
|
|
|
if (prc)
|
|
dest_rect = *prc;
|
|
else
|
|
{
|
|
dest_rect.X = dest_rect.Y = 0;
|
|
dest_rect.Width = This->width;
|
|
dest_rect.Height = This->height;
|
|
}
|
|
|
|
if (dest_rect.X < 0 || dest_rect.Y < 0 ||
|
|
dest_rect.X+dest_rect.Width > This->width|| dest_rect.Y+dest_rect.Height > This->height)
|
|
{
|
|
hr = E_INVALIDARG;
|
|
goto end;
|
|
}
|
|
|
|
bytesperrow = ((This->bpp * dest_rect.Width)+7)/8;
|
|
|
|
if (cbStride < bytesperrow)
|
|
{
|
|
hr = E_INVALIDARG;
|
|
goto end;
|
|
}
|
|
|
|
if ((cbStride * dest_rect.Height) > cbBufferSize)
|
|
{
|
|
hr = E_INVALIDARG;
|
|
goto end;
|
|
}
|
|
|
|
/* MSDN recommends calling CopyPixels once for each scanline from top to
|
|
* bottom, and claims codecs optimize for this. Ideally, when called in this
|
|
* way, we should avoid requesting a scanline from the source more than
|
|
* once, by saving the data that will be useful for the next scanline after
|
|
* the call returns. The GetRequiredSourceRect/CopyScanline functions are
|
|
* designed to make it possible to do this in a generic way, but for now we
|
|
* just grab all the data we need in each call. */
|
|
|
|
This->fn_get_required_source_rect(This, dest_rect.X, dest_rect.Y, &src_rect_ul);
|
|
This->fn_get_required_source_rect(This, dest_rect.X+dest_rect.Width-1,
|
|
dest_rect.Y+dest_rect.Height-1, &src_rect_br);
|
|
|
|
src_rect.X = src_rect_ul.X;
|
|
src_rect.Y = src_rect_ul.Y;
|
|
src_rect.Width = src_rect_br.Width + src_rect_br.X - src_rect_ul.X;
|
|
src_rect.Height = src_rect_br.Height + src_rect_br.Y - src_rect_ul.Y;
|
|
|
|
src_bytesperrow = (src_rect.Width * This->bpp + 7)/8;
|
|
buffer_size = src_bytesperrow * src_rect.Height;
|
|
|
|
src_rows = HeapAlloc(GetProcessHeap(), 0, sizeof(BYTE*) * src_rect.Height);
|
|
src_bits = HeapAlloc(GetProcessHeap(), 0, buffer_size);
|
|
|
|
if (!src_rows || !src_bits)
|
|
{
|
|
HeapFree(GetProcessHeap(), 0, src_rows);
|
|
HeapFree(GetProcessHeap(), 0, src_bits);
|
|
hr = E_OUTOFMEMORY;
|
|
goto end;
|
|
}
|
|
|
|
for (y=0; y<src_rect.Height; y++)
|
|
src_rows[y] = src_bits + y * src_bytesperrow;
|
|
|
|
hr = IWICBitmapSource_CopyPixels(This->source, &src_rect, src_bytesperrow,
|
|
buffer_size, src_bits);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
for (y=0; y < dest_rect.Height; y++)
|
|
{
|
|
This->fn_copy_scanline(This, dest_rect.X, dest_rect.Y+y, dest_rect.Width,
|
|
src_rows, src_rect.X, src_rect.Y, pbBuffer + cbStride * y);
|
|
}
|
|
}
|
|
|
|
HeapFree(GetProcessHeap(), 0, src_rows);
|
|
HeapFree(GetProcessHeap(), 0, src_bits);
|
|
|
|
end:
|
|
LeaveCriticalSection(&This->lock);
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI BitmapScaler_Initialize(IWICBitmapScaler *iface,
|
|
IWICBitmapSource *pISource, UINT uiWidth, UINT uiHeight,
|
|
WICBitmapInterpolationMode mode)
|
|
{
|
|
BitmapScaler *This = impl_from_IWICBitmapScaler(iface);
|
|
HRESULT hr;
|
|
GUID src_pixelformat;
|
|
|
|
TRACE("(%p,%p,%u,%u,%u)\n", iface, pISource, uiWidth, uiHeight, mode);
|
|
|
|
EnterCriticalSection(&This->lock);
|
|
|
|
if (This->source)
|
|
{
|
|
hr = WINCODEC_ERR_WRONGSTATE;
|
|
goto end;
|
|
}
|
|
|
|
This->width = uiWidth;
|
|
This->height = uiHeight;
|
|
This->mode = mode;
|
|
|
|
hr = IWICBitmapSource_GetSize(pISource, &This->src_width, &This->src_height);
|
|
|
|
if (SUCCEEDED(hr))
|
|
hr = IWICBitmapSource_GetPixelFormat(pISource, &src_pixelformat);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = get_pixelformat_bpp(&src_pixelformat, &This->bpp);
|
|
}
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
switch (mode)
|
|
{
|
|
default:
|
|
FIXME("unsupported mode %i\n", mode);
|
|
/* fall-through */
|
|
case WICBitmapInterpolationModeNearestNeighbor:
|
|
if ((This->bpp % 8) == 0)
|
|
{
|
|
IWICBitmapSource_AddRef(pISource);
|
|
This->source = pISource;
|
|
}
|
|
else
|
|
{
|
|
hr = WICConvertBitmapSource(&GUID_WICPixelFormat32bppBGRA,
|
|
pISource, &This->source);
|
|
This->bpp = 32;
|
|
}
|
|
This->fn_get_required_source_rect = NearestNeighbor_GetRequiredSourceRect;
|
|
This->fn_copy_scanline = NearestNeighbor_CopyScanline;
|
|
break;
|
|
}
|
|
}
|
|
|
|
end:
|
|
LeaveCriticalSection(&This->lock);
|
|
|
|
return hr;
|
|
}
|
|
|
|
static const IWICBitmapScalerVtbl BitmapScaler_Vtbl = {
|
|
BitmapScaler_QueryInterface,
|
|
BitmapScaler_AddRef,
|
|
BitmapScaler_Release,
|
|
BitmapScaler_GetSize,
|
|
BitmapScaler_GetPixelFormat,
|
|
BitmapScaler_GetResolution,
|
|
BitmapScaler_CopyPalette,
|
|
BitmapScaler_CopyPixels,
|
|
BitmapScaler_Initialize
|
|
};
|
|
|
|
HRESULT BitmapScaler_Create(IWICBitmapScaler **scaler)
|
|
{
|
|
BitmapScaler *This;
|
|
|
|
This = HeapAlloc(GetProcessHeap(), 0, sizeof(BitmapScaler));
|
|
if (!This) return E_OUTOFMEMORY;
|
|
|
|
This->IWICBitmapScaler_iface.lpVtbl = &BitmapScaler_Vtbl;
|
|
This->ref = 1;
|
|
This->source = NULL;
|
|
This->width = 0;
|
|
This->height = 0;
|
|
This->src_width = 0;
|
|
This->src_height = 0;
|
|
This->mode = 0;
|
|
This->bpp = 0;
|
|
InitializeCriticalSection(&This->lock);
|
|
This->lock.DebugInfo->Spare[0] = (DWORD_PTR)(__FILE__ ": BitmapScaler.lock");
|
|
|
|
*scaler = &This->IWICBitmapScaler_iface;
|
|
|
|
return S_OK;
|
|
}
|