Skip to content
Projects
Groups
Snippets
Help
This project
Loading...
Sign in / Register
Toggle navigation
W
wine-cw
Project
Project
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Registry
Registry
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
wine
wine-cw
Commits
07a6b89f
Commit
07a6b89f
authored
Jul 06, 2009
by
Vincent Povirk
Committed by
Alexandre Julliard
Jul 07, 2009
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
windowscodecs: Add support for decoding RLE8-encoded BMP files.
parent
387b8427
Show whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
291 additions
and
0 deletions
+291
-0
bmpdecode.c
dlls/windowscodecs/bmpdecode.c
+111
-0
bmpformat.c
dlls/windowscodecs/tests/bmpformat.c
+180
-0
No files found.
dlls/windowscodecs/bmpdecode.c
View file @
07a6b89f
...
...
@@ -386,6 +386,112 @@ fail:
return
hr
;
}
static
HRESULT
BmpFrameDecode_ReadRLE8
(
BmpFrameDecode
*
This
)
{
UINT
bytesperrow
;
UINT
width
,
height
;
BYTE
*
rledata
,
*
cursor
,
*
rledataend
;
UINT
rlesize
,
datasize
,
palettesize
;
DWORD
palette
[
256
];
UINT
x
,
y
;
DWORD
*
bgrdata
;
HRESULT
hr
;
LARGE_INTEGER
offbits
;
ULONG
bytesread
;
width
=
This
->
bih
.
bV5Width
;
height
=
abs
(
This
->
bih
.
bV5Height
);
bytesperrow
=
width
*
4
;
datasize
=
bytesperrow
*
height
;
rlesize
=
This
->
bih
.
bV5SizeImage
;
if
(
This
->
bih
.
bV5ClrUsed
&&
This
->
bih
.
bV5ClrUsed
<
256
)
palettesize
=
4
*
This
->
bih
.
bV5ClrUsed
;
else
palettesize
=
4
*
256
;
rledata
=
HeapAlloc
(
GetProcessHeap
(),
0
,
rlesize
);
This
->
imagedata
=
HeapAlloc
(
GetProcessHeap
(),
0
,
datasize
);
if
(
!
This
->
imagedata
||
!
rledata
)
{
hr
=
E_OUTOFMEMORY
;
goto
fail
;
}
/* read palette */
offbits
.
QuadPart
=
sizeof
(
BITMAPFILEHEADER
)
+
This
->
bih
.
bV5Size
;
hr
=
IStream_Seek
(
This
->
stream
,
offbits
,
STREAM_SEEK_SET
,
NULL
);
if
(
FAILED
(
hr
))
goto
fail
;
hr
=
IStream_Read
(
This
->
stream
,
palette
,
palettesize
,
&
bytesread
);
if
(
FAILED
(
hr
)
||
bytesread
!=
palettesize
)
goto
fail
;
/* read RLE data */
offbits
.
QuadPart
=
This
->
bfh
.
bfOffBits
;
hr
=
IStream_Seek
(
This
->
stream
,
offbits
,
STREAM_SEEK_SET
,
NULL
);
if
(
FAILED
(
hr
))
goto
fail
;
hr
=
IStream_Read
(
This
->
stream
,
rledata
,
rlesize
,
&
bytesread
);
if
(
FAILED
(
hr
)
||
bytesread
!=
rlesize
)
goto
fail
;
/* decode RLE */
bgrdata
=
(
DWORD
*
)
This
->
imagedata
;
x
=
0
;
y
=
0
;
rledataend
=
rledata
+
rlesize
;
cursor
=
rledata
;
while
(
cursor
<
rledataend
&&
y
<
height
)
{
BYTE
length
=
*
cursor
++
;
if
(
length
==
0
)
{
/* escape code */
BYTE
escape
=
*
cursor
++
;
switch
(
escape
)
{
case
0
:
/* end of line */
x
=
0
;
y
++
;
break
;
case
1
:
/* end of bitmap */
goto
end
;
case
2
:
/* delta */
if
(
cursor
<
rledataend
)
{
x
+=
*
cursor
++
;
y
+=
*
cursor
++
;
}
break
;
default:
/* absolute mode */
length
=
escape
;
while
(
cursor
<
rledataend
&&
length
--
&&
x
<
width
)
bgrdata
[
y
*
width
+
x
++
]
=
palette
[
*
cursor
++
];
if
(
escape
&
1
)
cursor
++
;
/* skip pad byte */
}
}
else
{
DWORD
color
=
palette
[
*
cursor
++
];
while
(
length
--
&&
x
<
width
)
bgrdata
[
y
*
width
+
x
++
]
=
color
;
}
}
end:
HeapFree
(
GetProcessHeap
(),
0
,
rledata
);
This
->
imagedatastart
=
This
->
imagedata
+
(
height
-
1
)
*
bytesperrow
;
This
->
stride
=
-
bytesperrow
;
return
S_OK
;
fail:
HeapFree
(
GetProcessHeap
(),
0
,
rledata
);
HeapFree
(
GetProcessHeap
(),
0
,
This
->
imagedata
);
This
->
imagedata
=
NULL
;
if
(
SUCCEEDED
(
hr
))
hr
=
E_FAIL
;
return
hr
;
}
static
HRESULT
BmpFrameDecode_ReadUnsupported
(
BmpFrameDecode
*
This
)
{
return
E_FAIL
;
...
...
@@ -515,6 +621,11 @@ static HRESULT BmpDecoder_ReadHeaders(BmpDecoder* This, IStream *stream)
FIXME
(
"unsupported bit depth %i for uncompressed RGB
\n
"
,
This
->
bih
.
bV5BitCount
);
}
break
;
case
BI_RLE8
:
This
->
bitsperpixel
=
32
;
This
->
read_data_func
=
BmpFrameDecode_ReadRLE8
;
This
->
pixelformat
=
&
GUID_WICPixelFormat32bppBGR
;
break
;
default:
This
->
bitsperpixel
=
0
;
This
->
read_data_func
=
BmpFrameDecode_ReadUnsupported
;
...
...
dlls/windowscodecs/tests/bmpformat.c
View file @
07a6b89f
...
...
@@ -542,6 +542,185 @@ static void test_decode_4bpp(void)
IWICBitmapDecoder_Release
(
decoder
);
}
static
const
char
testbmp_rle8
[]
=
{
/* BITMAPFILEHEADER */
66
,
77
,
/* "BM" */
202
,
0
,
0
,
0
,
/* file size */
0
,
0
,
0
,
0
,
/* reserved */
122
,
0
,
0
,
0
,
/* offset to bits */
/* BITMAPINFOHEADER */
40
,
0
,
0
,
0
,
/* header size */
8
,
0
,
0
,
0
,
/* width */
8
,
0
,
0
,
0
,
/* height */
1
,
0
,
/* planes */
8
,
0
,
/* bit count */
1
,
0
,
0
,
0
,
/* compression = BI_RLE8 */
80
,
0
,
0
,
0
,
/* image size */
19
,
11
,
0
,
0
,
/* X pixels per meter */
19
,
11
,
0
,
0
,
/* Y pixels per meter */
17
,
0
,
0
,
0
,
/* colors used */
17
,
0
,
0
,
0
,
/* colors important */
/* color table */
0
,
0
,
0
,
0
,
17
,
17
,
17
,
0
,
255
,
0
,
0
,
0
,
34
,
34
,
34
,
0
,
0
,
0
,
204
,
0
,
0
,
0
,
221
,
0
,
0
,
0
,
238
,
0
,
51
,
51
,
51
,
0
,
0
,
0
,
255
,
0
,
68
,
68
,
68
,
0
,
255
,
0
,
255
,
0
,
85
,
85
,
85
,
0
,
0
,
204
,
0
,
0
,
0
,
221
,
0
,
0
,
0
,
238
,
0
,
0
,
0
,
255
,
0
,
0
,
255
,
255
,
255
,
0
,
/* bits */
4
,
15
,
0
,
4
,
11
,
9
,
9
,
0
,
0
,
0
,
4
,
14
,
0
,
4
,
3
,
10
,
10
,
7
,
0
,
0
,
4
,
13
,
0
,
4
,
3
,
10
,
10
,
7
,
0
,
0
,
4
,
12
,
0
,
4
,
0
,
1
,
1
,
11
,
0
,
0
,
0
,
4
,
16
,
2
,
16
,
2
,
4
,
4
,
0
,
0
,
0
,
4
,
2
,
16
,
2
,
16
,
4
,
5
,
0
,
0
,
0
,
4
,
16
,
2
,
16
,
2
,
4
,
6
,
0
,
0
,
0
,
4
,
2
,
16
,
2
,
16
,
4
,
8
,
0
,
1
};
static
void
test_decode_rle8
(
void
)
{
IWICBitmapDecoder
*
decoder
,
*
decoder2
;
IWICBitmapFrameDecode
*
framedecode
;
HRESULT
hr
;
HGLOBAL
hbmpdata
;
char
*
bmpdata
;
IStream
*
bmpstream
;
DWORD
capability
=
0
;
GUID
guidresult
;
UINT
count
=
0
,
width
=
0
,
height
=
0
;
double
dpiX
,
dpiY
;
DWORD
imagedata
[
64
]
=
{
1
};
const
DWORD
expected_imagedata
[
64
]
=
{
0x0000ff
,
0xffffff
,
0x0000ff
,
0xffffff
,
0xff0000
,
0xff0000
,
0xff0000
,
0xff0000
,
0xffffff
,
0x0000ff
,
0xffffff
,
0x0000ff
,
0xee0000
,
0xee0000
,
0xee0000
,
0xee0000
,
0x0000ff
,
0xffffff
,
0x0000ff
,
0xffffff
,
0xdd0000
,
0xdd0000
,
0xdd0000
,
0xdd0000
,
0xffffff
,
0x0000ff
,
0xffffff
,
0x0000ff
,
0xcc0000
,
0xcc0000
,
0xcc0000
,
0xcc0000
,
0x00cc00
,
0x00cc00
,
0x00cc00
,
0x00cc00
,
0x000000
,
0x111111
,
0x111111
,
0x555555
,
0x00dd00
,
0x00dd00
,
0x00dd00
,
0x00dd00
,
0x222222
,
0xff00ff
,
0xff00ff
,
0x333333
,
0x00ee00
,
0x00ee00
,
0x00ee00
,
0x00ee00
,
0x222222
,
0xff00ff
,
0xff00ff
,
0x333333
,
0x00ff00
,
0x00ff00
,
0x00ff00
,
0x00ff00
,
0x555555
,
0x444444
,
0x444444
,
0x000000
};
WICColor
palettedata
[
17
]
=
{
1
};
const
WICColor
expected_palettedata
[
17
]
=
{
0xff000000
,
0xff111111
,
0xff0000ff
,
0xff222222
,
0xffcc0000
,
0xffdd0000
,
0xffee0000
,
0xff333333
,
0xffff0000
,
0xff444444
,
0xffff00ff
,
0xff555555
,
0xff00cc00
,
0xff00dd00
,
0xff00ee00
,
0xff00ff00
,
0xffffffff
};
WICRect
rc
;
hr
=
CoCreateInstance
(
&
CLSID_WICBmpDecoder
,
NULL
,
CLSCTX_INPROC_SERVER
,
&
IID_IWICBitmapDecoder
,
(
void
**
)
&
decoder
);
ok
(
SUCCEEDED
(
hr
),
"CoCreateInstance failed, hr=%x
\n
"
,
hr
);
if
(
!
SUCCEEDED
(
hr
))
return
;
hbmpdata
=
GlobalAlloc
(
GMEM_MOVEABLE
,
sizeof
(
testbmp_rle8
));
ok
(
hbmpdata
!=
0
,
"GlobalAlloc failed
\n
"
);
if
(
hbmpdata
)
{
bmpdata
=
GlobalLock
(
hbmpdata
);
memcpy
(
bmpdata
,
testbmp_rle8
,
sizeof
(
testbmp_rle8
));
GlobalUnlock
(
hbmpdata
);
hr
=
CreateStreamOnHGlobal
(
hbmpdata
,
FALSE
,
&
bmpstream
);
ok
(
SUCCEEDED
(
hr
),
"CreateStreamOnHGlobal failed, hr=%x
\n
"
,
hr
);
if
(
SUCCEEDED
(
hr
))
{
hr
=
IWICBitmapDecoder_Initialize
(
decoder
,
bmpstream
,
WICDecodeMetadataCacheOnLoad
);
ok
(
hr
==
S_OK
,
"Initialize failed, hr=%x
\n
"
,
hr
);
hr
=
IWICBitmapDecoder_GetContainerFormat
(
decoder
,
&
guidresult
);
ok
(
SUCCEEDED
(
hr
),
"GetContainerFormat failed, hr=%x
\n
"
,
hr
);
ok
(
IsEqualGUID
(
&
guidresult
,
&
GUID_ContainerFormatBmp
),
"unexpected container format
\n
"
);
hr
=
IWICBitmapDecoder_GetFrameCount
(
decoder
,
&
count
);
ok
(
SUCCEEDED
(
hr
),
"GetFrameCount failed, hr=%x
\n
"
,
hr
);
ok
(
count
==
1
,
"unexpected count %u
\n
"
,
count
);
hr
=
IWICBitmapDecoder_GetFrame
(
decoder
,
0
,
&
framedecode
);
ok
(
SUCCEEDED
(
hr
),
"GetFrame failed, hr=%x
\n
"
,
hr
);
if
(
SUCCEEDED
(
hr
))
{
IWICImagingFactory
*
factory
;
IWICPalette
*
palette
;
hr
=
IWICBitmapFrameDecode_GetSize
(
framedecode
,
&
width
,
&
height
);
ok
(
SUCCEEDED
(
hr
),
"GetSize failed, hr=%x
\n
"
,
hr
);
ok
(
width
==
8
,
"expected width=8, got %u
\n
"
,
width
);
ok
(
height
==
8
,
"expected height=8, got %u
\n
"
,
height
);
hr
=
IWICBitmapFrameDecode_GetResolution
(
framedecode
,
&
dpiX
,
&
dpiY
);
ok
(
SUCCEEDED
(
hr
),
"GetResolution failed, hr=%x
\n
"
,
hr
);
ok
(
fabs
(
dpiX
-
72
.
0
)
<
0
.
01
,
"expected dpiX=96.0, got %f
\n
"
,
dpiX
);
ok
(
fabs
(
dpiY
-
72
.
0
)
<
0
.
01
,
"expected dpiY=96.0, got %f
\n
"
,
dpiY
);
hr
=
IWICBitmapFrameDecode_GetPixelFormat
(
framedecode
,
&
guidresult
);
ok
(
SUCCEEDED
(
hr
),
"GetPixelFormat failed, hr=%x
\n
"
,
hr
);
ok
(
IsEqualGUID
(
&
guidresult
,
&
GUID_WICPixelFormat32bppBGR
),
"unexpected pixel format
\n
"
);
hr
=
CoCreateInstance
(
&
CLSID_WICImagingFactory
,
NULL
,
CLSCTX_INPROC_SERVER
,
&
IID_IWICImagingFactory
,
(
void
**
)
&
factory
);
ok
(
SUCCEEDED
(
hr
),
"CoCreateInstance failed, hr=%x
\n
"
,
hr
);
if
(
SUCCEEDED
(
hr
))
{
hr
=
IWICImagingFactory_CreatePalette
(
factory
,
&
palette
);
ok
(
SUCCEEDED
(
hr
),
"CreatePalette failed, hr=%x
\n
"
,
hr
);
if
(
SUCCEEDED
(
hr
))
{
hr
=
IWICBitmapDecoder_CopyPalette
(
decoder
,
palette
);
ok
(
hr
==
WINCODEC_ERR_PALETTEUNAVAILABLE
,
"expected WINCODEC_ERR_PALETTEUNAVAILABLE, got %x
\n
"
,
hr
);
hr
=
IWICBitmapFrameDecode_CopyPalette
(
framedecode
,
palette
);
ok
(
SUCCEEDED
(
hr
),
"CopyPalette failed, hr=%x
\n
"
,
hr
);
hr
=
IWICPalette_GetColorCount
(
palette
,
&
count
);
ok
(
SUCCEEDED
(
hr
),
"GetColorCount failed, hr=%x
\n
"
,
hr
);
ok
(
count
==
17
,
"expected count=17, got %u
\n
"
,
count
);
hr
=
IWICPalette_GetColors
(
palette
,
17
,
palettedata
,
&
count
);
ok
(
SUCCEEDED
(
hr
),
"GetColorCount failed, hr=%x
\n
"
,
hr
);
ok
(
count
==
17
,
"expected count=17, got %u
\n
"
,
count
);
ok
(
!
memcmp
(
palettedata
,
expected_palettedata
,
sizeof
(
palettedata
)),
"unexpected palette data
\n
"
);
IWICPalette_Release
(
palette
);
}
IWICImagingFactory_Release
(
factory
);
}
rc
.
X
=
0
;
rc
.
Y
=
0
;
rc
.
Width
=
8
;
rc
.
Height
=
8
;
hr
=
IWICBitmapFrameDecode_CopyPixels
(
framedecode
,
&
rc
,
32
,
sizeof
(
imagedata
),
(
BYTE
*
)
imagedata
);
ok
(
SUCCEEDED
(
hr
),
"CopyPixels failed, hr=%x
\n
"
,
hr
);
ok
(
!
memcmp
(
imagedata
,
expected_imagedata
,
sizeof
(
imagedata
)),
"unexpected image data
\n
"
);
IWICBitmapFrameDecode_Release
(
framedecode
);
}
hr
=
CoCreateInstance
(
&
CLSID_WICBmpDecoder
,
NULL
,
CLSCTX_INPROC_SERVER
,
&
IID_IWICBitmapDecoder
,
(
void
**
)
&
decoder2
);
ok
(
SUCCEEDED
(
hr
),
"CoCreateInstance failed, hr=%x
\n
"
,
hr
);
if
(
SUCCEEDED
(
hr
))
{
hr
=
IWICBitmapDecoder_QueryCapability
(
decoder2
,
bmpstream
,
&
capability
);
ok
(
hr
==
S_OK
,
"QueryCapability failed, hr=%x
\n
"
,
hr
);
ok
(
capability
==
(
WICBitmapDecoderCapabilityCanDecodeAllImages
),
"unexpected capabilities: %x
\n
"
,
capability
);
}
IStream_Release
(
bmpstream
);
}
GlobalFree
(
hbmpdata
);
}
IWICBitmapDecoder_Release
(
decoder
);
}
START_TEST
(
bmpformat
)
{
CoInitializeEx
(
NULL
,
COINIT_APARTMENTTHREADED
);
...
...
@@ -549,6 +728,7 @@ START_TEST(bmpformat)
test_decode_24bpp
();
test_decode_1bpp
();
test_decode_4bpp
();
test_decode_rle8
();
CoUninitialize
();
}
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment