2013-12-30 11:53:18 +01:00
|
|
|
/*
|
|
|
|
* Copyright 2013 Jacek Caban 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
|
|
|
|
*/
|
|
|
|
|
|
|
|
#define WIN32_LEAN_AND_MEAN
|
|
|
|
#include <windows.h>
|
|
|
|
#include <vfw.h>
|
|
|
|
#include <aviriff.h>
|
2014-01-02 14:16:20 +01:00
|
|
|
#include <stdio.h>
|
2013-12-30 11:53:18 +01:00
|
|
|
|
|
|
|
#include "wine/test.h"
|
|
|
|
|
2014-01-02 14:16:49 +01:00
|
|
|
static void test_output(const BYTE *output, int out_size, const BYTE *expect, int size)
|
|
|
|
{
|
|
|
|
char buf[512], *ptr;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
i = out_size == size && !memcmp(output, expect, size);
|
|
|
|
ok(i, "Unexpected output\n");
|
|
|
|
if(i)
|
|
|
|
return;
|
|
|
|
|
|
|
|
for(i=0, ptr=buf; i<out_size; i++)
|
|
|
|
ptr += sprintf(ptr, "%x ", output[i]);
|
|
|
|
trace("Got: %s\n", buf);
|
|
|
|
for(i=0, ptr=buf; i<size; i++)
|
|
|
|
ptr += sprintf(ptr, "%x ", expect[i]);
|
|
|
|
trace("Exp: %s\n", buf);
|
|
|
|
}
|
|
|
|
|
2013-12-30 11:53:18 +01:00
|
|
|
static void test_encode(void)
|
|
|
|
{
|
2014-01-02 14:16:49 +01:00
|
|
|
BITMAPINFOHEADER *output_header;
|
|
|
|
DWORD output_size, flags, quality;
|
|
|
|
BYTE buf[64];
|
2013-12-30 11:53:18 +01:00
|
|
|
ICINFO info;
|
|
|
|
HIC hic;
|
|
|
|
LRESULT res;
|
|
|
|
|
2014-01-02 14:16:49 +01:00
|
|
|
struct { BITMAPINFOHEADER header; RGBQUAD map[256]; }
|
|
|
|
input_header = { {sizeof(BITMAPINFOHEADER), 32, 1, 1, 8, 0, 32*8, 0, 0, 256, 256},
|
|
|
|
{{255,0,0}, {0,255,0}, {0,0,255}, {255,255,255}}};
|
|
|
|
|
|
|
|
static BYTE input1[32] = {1,2,3,3,3,3,2,3,1};
|
|
|
|
static const BYTE output1[] = {1,1,1,2,4,3,0,3,2,3,1,0,23,0,0,0,0,1};
|
|
|
|
|
2013-12-30 11:53:18 +01:00
|
|
|
hic = ICOpen(FCC('V','I','D','C'), FCC('m','r','l','e'), ICMODE_COMPRESS);
|
|
|
|
ok(hic != NULL, "ICOpen failed\n");
|
|
|
|
|
|
|
|
res = ICGetInfo(hic, &info, sizeof(info));
|
|
|
|
ok(res == sizeof(info), "res = %ld\n", res);
|
|
|
|
ok(info.dwSize == sizeof(info), "dwSize = %d\n", info.dwSize);
|
|
|
|
ok(info.fccHandler == FCC('M','R','L','E'), "fccHandler = %x\n", info.fccHandler);
|
|
|
|
todo_wine ok(info.dwFlags == (VIDCF_QUALITY|VIDCF_CRUNCH|VIDCF_TEMPORAL), "dwFlags = %x\n", info.dwFlags);
|
|
|
|
ok(info.dwVersionICM == ICVERSION, "dwVersionICM = %d\n", info.dwVersionICM);
|
|
|
|
|
2014-01-02 14:16:20 +01:00
|
|
|
quality = 0xdeadbeef;
|
|
|
|
res = ICSendMessage(hic, ICM_GETDEFAULTQUALITY, (DWORD_PTR)&quality, 0);
|
|
|
|
ok(res == ICERR_OK, "ICSendMessage(ICM_GETDEFAULTQUALITY) failed: %ld\n", res);
|
|
|
|
ok(quality == 8500, "quality = %d\n", quality);
|
|
|
|
|
2014-01-02 14:16:31 +01:00
|
|
|
quality = 0xdeadbeef;
|
|
|
|
res = ICSendMessage(hic, ICM_GETQUALITY, (DWORD_PTR)&quality, 0);
|
|
|
|
ok(res == ICERR_UNSUPPORTED, "ICSendMessage(ICM_GETQUALITY) failed: %ld\n", res);
|
|
|
|
ok(quality == 0xdeadbeef, "quality = %d\n", quality);
|
|
|
|
|
|
|
|
quality = ICQUALITY_HIGH;
|
|
|
|
res = ICSendMessage(hic, ICM_SETQUALITY, (DWORD_PTR)&quality, 0);
|
|
|
|
ok(res == ICERR_UNSUPPORTED, "ICSendMessage(ICM_SETQUALITY) failed: %ld\n", res);
|
|
|
|
|
2014-01-02 14:16:49 +01:00
|
|
|
output_size = ICCompressGetFormatSize(hic, &input_header.header);
|
|
|
|
ok(output_size == 1064, "output_size = %d\n", output_size);
|
|
|
|
|
|
|
|
output_header = HeapAlloc(GetProcessHeap(), 0, output_size);
|
|
|
|
ICCompressGetFormat(hic, &input_header.header, output_header);
|
|
|
|
|
|
|
|
flags = 0;
|
|
|
|
res = ICCompress(hic, ICCOMPRESS_KEYFRAME, output_header, buf, &input_header.header, input1, 0, &flags, 0, 0, 0, NULL, NULL);
|
|
|
|
ok(res == ICERR_OK, "ICCompress failed: %ld\n", res);
|
|
|
|
test_output(buf, output_header->biSizeImage, output1, sizeof(output1));
|
2014-01-02 14:17:11 +01:00
|
|
|
ok(flags == (AVIIF_TWOCC|AVIIF_KEYFRAME), "flags = %x\n", flags);
|
2014-01-02 14:16:49 +01:00
|
|
|
|
|
|
|
HeapFree(GetProcessHeap(), 0, output_header);
|
|
|
|
|
2013-12-30 11:53:18 +01:00
|
|
|
ICClose(hic);
|
|
|
|
}
|
|
|
|
|
2015-11-05 14:01:22 +01:00
|
|
|
static void test_raw_decompress(void)
|
|
|
|
{
|
|
|
|
DWORD codecs[] = {FCC('D', 'I', 'B', ' '), FCC('R', 'A', 'W', ' '),
|
|
|
|
FCC('M', 'R', 'L', 'E'), BI_RGB}, i, hr;
|
|
|
|
BITMAPINFO bi;
|
|
|
|
BITMAPINFOHEADER *bih, biho;
|
|
|
|
HIC hic;
|
|
|
|
ICINFO codec_info;
|
|
|
|
void *bits, *outbits;
|
|
|
|
|
|
|
|
/* Create an uncompressed 200x200 bitmap */
|
|
|
|
bih = &bi.bmiHeader;
|
|
|
|
bih->biSize = sizeof(*bih);
|
|
|
|
bih->biWidth = 200;
|
|
|
|
bih->biHeight = 200;
|
|
|
|
bih->biPlanes = 1;
|
|
|
|
bih->biBitCount = 24;
|
|
|
|
bih->biCompression = BI_RGB;
|
|
|
|
bih->biSizeImage = bih->biWidth * (bih->biBitCount / 8) * bih->biHeight;
|
|
|
|
bih->biXPelsPerMeter = 10000;
|
|
|
|
bih->biYPelsPerMeter = 10000;
|
|
|
|
bih->biClrUsed = 0;
|
|
|
|
bih->biClrImportant = 0;
|
|
|
|
biho = *bih;
|
|
|
|
|
|
|
|
bits = HeapAlloc(GetProcessHeap(), 0, bih->biSizeImage);
|
|
|
|
ok(bits != NULL, "Expected non-NULL value\n");
|
|
|
|
outbits = HeapAlloc(GetProcessHeap(), 0, bih->biSizeImage);
|
|
|
|
ok(outbits != NULL, "Expected non-NULL value\n");
|
|
|
|
|
|
|
|
for (i = 0; i < sizeof(codecs) / sizeof(codecs[0]); i++)
|
|
|
|
{
|
|
|
|
memset(bits, i + 0xAF, bih->biSizeImage);
|
2015-11-05 14:01:39 +01:00
|
|
|
|
2015-11-05 14:01:22 +01:00
|
|
|
/* Check which codec is able to decompress uncompressed data */
|
|
|
|
hic = ICLocate(FCC('V', 'I', 'D', 'C'), codecs[i], bih, NULL, ICMODE_DECOMPRESS);
|
|
|
|
ok(hic != NULL, "Test[%d]: Expected non-NULL return\n", i);
|
|
|
|
|
2015-12-02 01:42:31 +01:00
|
|
|
/* Now which is this codec? Windows returns MRLE for uncompressed cases */
|
2015-11-05 14:01:22 +01:00
|
|
|
memset(&codec_info, 0, sizeof(codec_info));
|
|
|
|
hr = ICGetInfo(hic, &codec_info, sizeof(codec_info));
|
|
|
|
ok(hr == sizeof(codec_info), "Test[%d]: Incorrect amount of data returned\n", i);
|
|
|
|
ok(codec_info.fccType == FCC('v', 'i', 'd', 'c'),
|
|
|
|
"Test[%d]: Expected a video type, got 0x%x\n", i, codec_info.fccType);
|
|
|
|
ok(codec_info.fccHandler == FCC('M', 'R', 'L', 'E'),
|
|
|
|
"Test[%d]: Expected MRLE, got 0x%x\n", i, codec_info.fccHandler);
|
|
|
|
|
|
|
|
/* Decompress the frame and check if we get the same output */
|
|
|
|
memset(outbits, 0, bih->biSizeImage);
|
|
|
|
hr = ICDecompress(hic, 0, bih, bits, &biho, outbits);
|
|
|
|
ok(hr == ICERR_OK, "Test[%d]: Expected ICERR_OK, got %d\n", i, hr);
|
|
|
|
ok(!memcmp(bits, outbits, bih->biSizeImage), "Test[%d]: Image contents do not match!\n", i);
|
|
|
|
|
|
|
|
hr = ICClose(hic);
|
|
|
|
ok(hr == ICERR_OK, "Test[%d]: Expected ICERR_OK, got %d\n", i, hr);
|
|
|
|
}
|
|
|
|
HeapFree(GetProcessHeap(), 0, bits);
|
|
|
|
HeapFree(GetProcessHeap(), 0, outbits);
|
|
|
|
}
|
|
|
|
|
2013-12-30 11:53:18 +01:00
|
|
|
START_TEST(msrle)
|
|
|
|
{
|
|
|
|
test_encode();
|
2015-11-05 14:01:22 +01:00
|
|
|
test_raw_decompress();
|
2013-12-30 11:53:18 +01:00
|
|
|
}
|