Windows2000/private/ntos/w32/ntuser/client/acons.c
2020-09-30 17:12:32 +02:00

984 lines
26 KiB
C

/****************************** Module Header ******************************\
* Module Name: acons.c
* Copyright (c) 1985 - 1999, Microsoft Corporation
* This module contains code for dealing with animated icons/cursors.
* History:
* 10-02-91 DarrinM Created.
* 07-30-92 DarrinM Unicodized.
* 11-28-94 JimA Moved to client from server.
*/
#include "precomp.h"
#pragma hdrstop
/*
* Resource Directory format for IconEditor generated icon and cursor
* (.ICO & .CUR) files. All fields are shared except xHotspot and yHotspot
* which are only valid for cursors.
*/
typedef struct _ICONFILERESDIR { // ird
BYTE bWidth;
BYTE bHeight;
BYTE bColorCount;
BYTE bReserved;
WORD xHotspot;
WORD yHotspot;
DWORD dwDIBSize;
DWORD dwDIBOffset;
} ICONFILERESDIR;
typedef struct _HOTSPOTREC { // hs
WORD xHotspot;
WORD yHotspot;
} HOTSPOTREC;
PCURSORRESOURCE ReadIconGuts(
IN PFILEINFO pfi,
IN LPNEWHEADER pnhBase,
IN int offResBase,
OUT LPWSTR *prt,
IN int cxDesired,
IN int cyDesired,
IN DWORD LR_flags);
BOOL ReadTag(
IN PFILEINFO pfi,
OUT PRTAG ptag);
BOOL ReadChunk(
IN PFILEINFO pfi,
IN PRTAG ptag,
OUT PVOID pv);
BOOL SkipChunk(
IN PFILEINFO pfi,
IN PRTAG ptag);
HICON CreateAniIcon(
LPCWSTR pszName,
LPWSTR rt,
int cicur,
DWORD *aicur,
int cpcur,
HCURSOR *ahcur,
JIF jifRate,
PJIF ajifRate,
BOOL fPublic);
HCURSOR ReadIconFromFileMap(
IN PFILEINFO pfi,
IN int cbSize,
IN DWORD cxDesired,
IN DWORD cyDesired,
IN DWORD LR_flags);
HICON LoadAniIcon(
IN PFILEINFO pfi,
IN LPWSTR rt,
IN DWORD cxDesired,
IN DWORD cyDesired,
IN DWORD LR_flags);
/*
* LoadCursorFromFile (API)
* Called by SetSystemCursor.
* History:
* 08-03-92 DarrinM Created.
*/
HCURSOR WINAPI LoadCursorFromFileW(
LPCWSTR pszFilename)
{
return(LoadImage(NULL,
pszFilename,
IMAGE_CURSOR,
0,
0,
LR_DEFAULTSIZE | LR_LOADFROMFILE));
}
/*
* LoadCursorFromFileA
* Returns: hCursor
* 10/9/1995 Created SanfordS
*/
HCURSOR WINAPI LoadCursorFromFileA(
LPCSTR pszFilename)
{
LPWSTR lpUniName;
HCURSOR hcur;
if (pszFilename == NULL ||
!MBToWCS(pszFilename, -1, &lpUniName, -1, TRUE))
return (HANDLE)NULL;
hcur = LoadCursorFromFileW(lpUniName);
UserLocalFree(lpUniName);
return hcur;
}
/*
* ReadFilePtr
* Works like ReadFile but with pointers to a mapped file buffer.
* Returns:
* 11/16/1995 Created SanfordS
*/
BOOL ReadFilePtr(
IN PFILEINFO pfi,
OUT LPVOID *ppBuf,
IN DWORD cb)
{
*ppBuf = pfi->pFilePtr;
pfi->pFilePtr += cb;
return (pfi->pFilePtr <= pfi->pFileEnd);
}
/*
* ReadFilePtrUnaligned
* Works like ReadFile but with pointers to a mapped file buffer.
* Returns:
* 11/16/1995 Created SanfordS
*/
BOOL ReadFilePtrUnaligned(
IN PFILEINFO pfi,
OUT VOID UNALIGNED **ppBuf,
IN DWORD cb)
{
*ppBuf = pfi->pFilePtr;
pfi->pFilePtr += cb;
return (pfi->pFilePtr <= pfi->pFileEnd);
}
/*
* ReadFilePtrCopy
* Works even more like ReadFile in that is copies data to the given buffer.
* Returns:
* 11/16/1995 Created SanfordS
*/
BOOL ReadFilePtrCopy(
IN PFILEINFO pfi,
IN OUT LPVOID pBuf,
IN DWORD cb)
{
if (pfi->pFilePtr + cb > pfi->pFileEnd) {
return(FALSE);
}
RtlCopyMemory(pBuf, pfi->pFilePtr, cb);
pfi->pFilePtr += cb;
return TRUE;
}
/*
* ReadTag, ReadChunk, SkipChunk
* Some handy functions for reading RIFF files.
* History:
* 10-02-91 DarrinM Created.
* 03-25-93 Jonpa Changed to use RIFF format instead of ASDF
*/
BOOL ReadTag(
IN PFILEINFO pfi,
OUT PRTAG ptag)
{
ptag->ckID = ptag->ckSize = 0L; // in case we fail the read.
return(ReadFilePtrCopy(pfi, ptag, sizeof(RTAG)));
}
BOOL ReadChunk(
IN PFILEINFO pfi,
IN PRTAG ptag,
OUT PVOID pv)
{
if (!ReadFilePtrCopy(pfi, pv, ptag->ckSize))
return FALSE;
/* WORD align file pointer */
if( ptag->ckSize & 1 )
pfi->pFilePtr++;
if (pfi->pFilePtr <= pfi->pFileEnd) {
return TRUE;
} else {
RIPMSG0(RIP_WARNING, "ReadChunk: Advanced pointer past end of file map");
return FALSE;
}
}
BOOL SkipChunk(
IN PFILEINFO pfi,
IN PRTAG ptag)
{
/*
* Round ptag->ckSize up to nearest word boundary
* to maintain alignment
*/
pfi->pFilePtr += (ptag->ckSize + 1) & (~1);
if (pfi->pFilePtr <= pfi->pFileEnd) {
return TRUE;
} else {
RIPMSG0(RIP_WARNING, "SkipChunk: Advanced pointer past end of file map");
return FALSE;
}
}
/*
* LoadCursorIconFromFileMap
* If pszName is one of the IDC_* values then we use WIN.INI to find a
* custom cursor/icon. Otherwise, pszName points to a filename of a .ICO/.CUR
* file to be loaded. If the file is an .ANI file containing a multiframe
* animation then LoadAniIcon is called to create an ACON. Otherwise if
* the file is an .ANI file containing just a single frame then it is loaded
* and a normal CURSOR/ICON resource is created from it.
* 12-26-91 DarrinM Wrote it.
* 03-17-93 JonPa Changed to use RIFF format for ani-cursors
* 11/16/1995 SanfordS Added LR_flags support
*/
HANDLE LoadCursorIconFromFileMap(
IN PFILEINFO pfi,
IN OUT LPWSTR *prt,
IN DWORD cxDesired,
IN DWORD cyDesired,
IN DWORD LR_flags,
OUT LPBOOL pfAni)
{
LPNEWHEADER pnh;
int offResBase;
*pfAni = FALSE;
offResBase = 0;
/*
* Determine if this is an .ICO/.CUR file or an .ANI file.
*/
pnh = (LPNEWHEADER)pfi->pFileMap;
if (*(LPDWORD)pnh == FOURCC_RIFF) {
RTAG tag;
/*
* It's an ANICURSOR!
* Seek back to beginning + 1 tag.
*/
pfi->pFilePtr = pfi->pFileMap + sizeof(tag);
/* check RIFF type for ACON */
if (*(LPDWORD)pfi->pFilePtr != FOURCC_ACON) {
return NULL;
}
pfi->pFilePtr += sizeof(DWORD);
if (pfi->pFilePtr > pfi->pFileEnd) {
return NULL;
}
/*
* Ok, we have a ACON chunk. Find the first ICON chunk and set
* things up so it looks we've just loaded the header of a normal
* .CUR file, then fall into the .CUR bits handling code below.
*/
while (ReadTag(pfi, &tag)) {
/*
* Handle each chunk type.
*/
if (tag.ckID == FOURCC_anih) {
ANIHEADER anih;
if (!ReadChunk(pfi, &tag, &anih)) {
return NULL;
}
if (!(anih.fl & AF_ICON) || (anih.cFrames == 0)) {
return NULL;
}
// If this ACON has more than one frame then go ahead
// and create an ACON, otherwise just use the first
// frame to create a normal ICON/CURSOR.
if (anih.cFrames > 1) {
*pfAni = TRUE;
*prt = RT_CURSOR;
return(LoadAniIcon(pfi,
RT_CURSOR,
cxDesired,
cyDesired,
LR_flags));
}
} else if (tag.ckID == FOURCC_LIST) {
LPDWORD pdwType = NULL;
BOOL fOK = FALSE;
/*
* If this is the fram list, then get the first icon out of it
*/
/* check LIST type for fram */
if( tag.ckSize >= sizeof(DWORD) &&
(fOK = ReadFilePtr( pfi,
&pdwType,
sizeof(DWORD))) &&
*pdwType == FOURCC_fram) {
if (!ReadTag(pfi, &tag)) {
return NULL;
}
if (tag.ckID == FOURCC_icon) {
/*
* We've found what we're looking for. Get current position
* in file to be used as the base from which the icon data
* offsets are offset from.
*/
offResBase = (int)(pfi->pFilePtr - pfi->pFileMap);
/*
* Grab the header first, since the following code assumes
* it was read above.
*/
ReadFilePtr(pfi, &pnh, sizeof(NEWHEADER));
/*
* Break out and let the icon loading/cursor creating code
* take it from here.
*/
break;
} else {
SkipChunk(pfi, &tag);
}
} else {
/*
* Something bad happened in the type read, if it was
* a file error then close and exit, otherwise just
* skip the rest of the chunk
*/
if(!fOK) {
return NULL;
}
/*
* take the type we just read out of the tag size and
* skip the rest
*/
tag.ckSize -= sizeof(DWORD);
SkipChunk(pfi, &tag);
}
} else {
/*
* We're not interested in this chunk, skip it.
*/
SkipChunk(pfi, &tag);
}
}
} else { // not a RIFF file.
if ((pnh->ResType != FT_ICON) && (pnh->ResType != FT_CURSOR)) {
return NULL;
}
}
{
PCURSORRESOURCE pcres;
pcres = ReadIconGuts(pfi,
pnh,
offResBase,
prt,
cxDesired,
cyDesired,
LR_flags);
return ConvertDIBIcon((LPBITMAPINFOHEADER)pcres,
NULL,
pfi->pszName,
*prt == RT_ICON,
cxDesired,
cyDesired,
LR_flags);
}
}
/*
* ReadIconGuts
* Returns: a pointer to a locally allocated buffer extraced from the
* given file that looks like a icon/acon resource.
* Also returns the type of the icon (RT_ICON or RT_CURSOR)
* 8/23/1995 SanfordS Documented
* 11/16/1995 SanfordS Added LR_flags support
*/
PCURSORRESOURCE ReadIconGuts(
IN PFILEINFO pfi,
IN NEWHEADER *pnhBase,
IN int offResBase,
OUT LPWSTR *prt,
IN int cxDesired,
IN int cyDesired,
IN DWORD LR_flags)
{
NEWHEADER *pnh;
int i, Id;
ICONFILERESDIR UNALIGNED *pird;
PCURSORRESOURCE pcres;
RESDIR UNALIGNED *prd;
DWORD cb;
HOTSPOTREC UNALIGNED *phs;
LPBITMAPINFOHEADER pbih;
/*
* Construct a fake array of RESDIR entries using the info at the head
* of the file. Store the data offset in the idIcon WORD so it can be
* returned by RtlGetIdFromDirectory.
*/
pnh = (NEWHEADER *)UserLocalAlloc(0, sizeof(NEWHEADER) +
(pnhBase->ResCount * (sizeof(RESDIR) + sizeof(HOTSPOTREC))));
if (pnh == NULL)
return NULL;
*pnh = *pnhBase;
prd = (RESDIR UNALIGNED *)(pnh + 1);
phs = (HOTSPOTREC UNALIGNED *)(prd + pnhBase->ResCount);
for (i = 0; i < (int)pnh->ResCount; i++, prd++) {
/*
* Read the resource directory from the icon file.
*/
ReadFilePtrUnaligned(pfi, &pird, sizeof(ICONFILERESDIR));
/*
* Convert from the icon editor's resource directory format
* to the post-RC.EXE format LookupIconIdFromDirectory expects.
*/
prd->Icon.Width = pird->bWidth;
prd->Icon.Height = pird->bHeight;
if (pnh->ResType == FT_ICON) { // ICON
prd->Icon.ColorCount = pird->bColorCount;
prd->Icon.reserved = 0;
}
prd->Planes = 0; // Hopefully nobody uses this
prd->BitCount = 0; // " "
prd->BytesInRes = pird->dwDIBSize;
prd->idIcon = (WORD)pird->dwDIBOffset;
phs->xHotspot = pird->xHotspot;
phs->yHotspot = pird->yHotspot;
phs++;
}
*prt = pnhBase->ResType == FT_ICON ? RT_ICON : RT_CURSOR;
Id = RtlGetIdFromDirectory((PBYTE)pnh,
*prt == RT_ICON,
cxDesired,
cyDesired,
LR_flags,
&cb);
/*
* Allocate for worst case (cursor).
*/
pcres = (PCURSORRESOURCE)UserLocalAlloc(0,
cb + FIELD_OFFSET(CURSORRESOURCE, bih));
if (pcres == NULL) {
goto CleanExit;
}
if (*prt == RT_CURSOR) {
/*
* Fill in hotspot info for cursors.
*/
prd = (RESDIR UNALIGNED *)(pnh + 1);
phs = (HOTSPOTREC UNALIGNED *)(prd + pnh->ResCount);
for( i = 0; i < pnh->ResCount; i++ ) {
if (prd[i].idIcon == (WORD)Id) {
pcres->xHotspot = phs[i].xHotspot;
pcres->yHotspot = phs[i].yHotspot;
break;
}
}
if (i == pnh->ResCount) {
pcres->xHotspot = pird->xHotspot;
pcres->yHotspot = pird->yHotspot;
}
pbih = &pcres->bih;
} else {
pbih = (LPBITMAPINFOHEADER)pcres;
}
/*
* Read in the header information into pcres.
*/
pfi->pFilePtr = pfi->pFileMap + offResBase + Id;
if (!ReadFilePtrCopy(pfi, pbih, cb)) {
UserLocalFree(pnh);
UserLocalFree(pcres);
return NULL;
}
CleanExit:
UserLocalFree(pnh);
return pcres;
}
/*
* CreateAniIcon
* For now, CreateAniIcon copies the jif rate table and the sequence table
* but not the CURSOR structs. This is ok as long as this routine is
* internal only.
* History:
* 10-02-91 DarrinM Created.
*/
HCURSOR CreateAniIcon(
LPCWSTR pszName,
LPWSTR rt,
int cicur,
DWORD *aicur,
int cpcur,
HCURSOR *ahcur,
JIF jifRate,
PJIF ajifRate,
BOOL fPublic)
{
HCURSOR hacon;
CURSORDATA acon;
DWORD cbacon;
HCURSOR *ahcurT; // Array of image frame pointers
DWORD *aicurT; // Array of frame indices (sequence table)
PJIF ajifRateT; // Array of time offsets
int i;
/*
* Start by allocating space for the ACON structure and the ahcur and
* ajifRate arrays.
*/
hacon = (HCURSOR)NtUserCallOneParam(fPublic,
SFI__CREATEEMPTYCURSOROBJECT);
if (hacon == NULL)
return NULL;
/*
* Save a couple LocalAlloc calls by allocating the memory needed for
* the CURSOR, JIF, and SEQ arrays at once.
*/
RtlZeroMemory(&acon, sizeof(acon));
cbacon = (cpcur * sizeof(HCURSOR)) +
(cicur * sizeof(JIF)) + (cicur * sizeof(DWORD));
ahcurT = (HCURSOR *)UserLocalAlloc(HEAP_ZERO_MEMORY, cbacon);
if (ahcurT == NULL) {
NtUserDestroyCursor((HCURSOR)hacon, CURSOR_ALWAYSDESTROY);
return NULL;
}
acon.aspcur = (PCURSOR *)ahcurT;
/*
* Set up work pointers
*/
ajifRateT = (PJIF)((PBYTE)ahcurT + (cpcur * sizeof(HCURSOR)));
aicurT = (DWORD *)((PBYTE)ajifRateT + (cicur * sizeof(JIF)));
/*
* Save offsets to arrays to make copying them to the server
* easier.
*/
acon.ajifRate = (PJIF)(cpcur * sizeof(HCURSOR));
acon.aicur = (DWORD *)((PBYTE)acon.ajifRate + (cicur * sizeof(JIF)));
acon.cpcur = cpcur;
acon.cicur = cicur;
acon.CURSORF_flags = CURSORF_ACON;
/*
* Store this information away so we can identify
* repeated calls to LoadCursor/Icon for the same
* resource type/id.
*/
acon.rt = PTR_TO_ID(rt);
acon.lpModName = szUSER32;
acon.lpName = (LPWSTR)pszName;
/*
* Make a private copy of the cursor pointers and the animation rate table.
*/
for (i = 0; i < cpcur; i++) {
ahcurT[i] = ahcur[i];
// ahcurT[i]->fPointer |= PTRI_ANIMATED; // if GDI needs it
}
for (i = 0; i < cicur; i++) {
/*
* If constant rate, initialize the rate table to a single value.
*/
if (ajifRate == NULL)
ajifRateT[i] = jifRate;
else
ajifRateT[i] = ajifRate[i];
/*
* If no sequence table then build a unity map to the cursor table.
*/
if (aicur == NULL)
aicurT[i] = i;
else
aicurT[i] = aicur[i];
}
/*
* Stuff acon data into the cursor
*/
if (!_SetCursorIconData(hacon, &acon)) {
NtUserDestroyCursor(hacon, CURSOR_ALWAYSDESTROY);
hacon = NULL;
}
UserLocalFree(ahcurT);
return hacon;
}
/*
* ReadIconFromFileMap
* LATER: Error handling.
* History:
* 12-21-91 DarrinM Created.
*/
HCURSOR ReadIconFromFileMap(
PFILEINFO pfi,
int cbSize, // used to seek past this chunk in case of error
DWORD cxDesired,
DWORD cyDesired,
DWORD LR_flags)
{
PCURSORRESOURCE pcres;
HCURSOR hcur = NULL;
LPNEWHEADER pnh;
int offResBase;
LPWSTR rt;
/*
* Get current position in file to be used as the base from which
* the icon data offsets are offset from.
*/
offResBase = (int)(pfi->pFilePtr - pfi->pFileMap);
/*
* Read the .ICO/.CUR data's header.
*/
ReadFilePtr(pfi, &pnh, sizeof(NEWHEADER));
pcres = ReadIconGuts(pfi,
pnh,
offResBase,
&rt,
cxDesired,
cyDesired,
LR_flags);
if (pcres != NULL) {
hcur = (HCURSOR)ConvertDIBIcon((LPBITMAPINFOHEADER)pcres,
NULL,
NULL,
(rt == RT_ICON),
cxDesired,
cyDesired,
LR_ACONFRAME | LR_flags);
UserLocalFree(pcres);
}
/*
* Seek to the end of this chunk, regardless of our current position.
*/
pfi->pFilePtr = pfi->pFileMap + ((offResBase + cbSize + 1) & (~1));
return hcur;
}
/*
* LoadAniIcon
* Loads an animatied cursor from a RIFF file. The RIFF file format for
* animated cursors looks like this:
* RIFF( 'ACON'
* LIST( 'INFO'
* INAM( <name> )
* IART( <artist> )
* )
* anih( <anihdr> )
* [rate( <rateinfo> ) ]
* ['seq '( <seq_info> )]
* LIST( 'fram' icon( <icon_file> ) ... )
* )
* History:
* 10-02-91 DarrinM Created.
* 03-17-93 JonPa Rewrote to use RIFF format instead of RAD
* 04-22-93 JonPa Finalized RIFF format (changed from ANI to ACON etc)
* 11/16/1995 SanfordS Added LR_flags support.
*/
HICON LoadAniIcon(
IN PFILEINFO pfi,
IN LPWSTR rt,
IN DWORD cxDesired,
IN DWORD cyDesired,
IN DWORD LR_flags)
{
int cpcur, ipcur = 0, i, cicur;
ANIHEADER anih;
ANIHEADER *panih = NULL;
HICON hacon = NULL;
HCURSOR *phcur;
JIF jifRate, *pjifRate;
RTAG tag;
DWORD *picur;
/*
* Position to the beginning of the file.
*/
pfi->pFilePtr = pfi->pFileMap + sizeof(tag);
#if DBG
if ((ULONG_PTR)pfi->pFileEnd != ((ULONG_PTR)(pfi->pFileMap + sizeof (RTAG) + ((RTAG *)(pfi->pFileMap))->ckSize + 1) & ~1)) {
RIPMSG2(RIP_WARNING, "LoadAniIcon: First RIFF chunk has invalid ckSize. Actual:%#lx Expected:%#lx",
((RTAG *)(pfi->pFileMap))->ckSize, (pfi->pFileEnd - pfi->pFileMap - sizeof(RTAG)) & ~1);
}
#endif
/* read the chunk type */
if(!ReadFilePtrCopy(pfi,
&tag.ckID,
sizeof(tag.ckID))) {
goto laiFileErr;
}
if (tag.ckID != FOURCC_ACON)
goto laiFileErr;
/* look for 'anih', 'rate', 'seq ', and 'icon' chunks */
while( ReadTag(pfi, &tag)) {
switch( tag.ckID ) {
case FOURCC_anih:
if (!ReadChunk(pfi, &tag, &anih))
goto laiFileErr;
if (!(anih.fl & AF_ICON) || (anih.cFrames == 0))
goto laiFileErr;
/*
* Allocate space for the ANIHEADER, HCURSOR array and a
* rate table (in case we run into one later).
*/
cpcur = anih.cFrames;
cicur = anih.cSteps;
panih = (PANIHEADER)UserLocalAlloc(HEAP_ZERO_MEMORY, sizeof(ANIHEADER) +
(cicur * sizeof(JIF)) + (cpcur * sizeof(HCURSOR)) +
(cicur * sizeof(DWORD)));
if (panih == NULL)
goto laiFileErr;
phcur = (HCURSOR *)((PBYTE)panih + sizeof(ANIHEADER));
pjifRate = NULL;
picur = NULL;
*panih = anih;
jifRate = panih->jifRate;
break;
case FOURCC_rate:
/*
* If we find a rate chunk, read it into its preallocated
* space.
*/
pjifRate = (PJIF)((PBYTE)phcur + cpcur * sizeof(HCURSOR));
if(!ReadChunk(pfi, &tag, (PBYTE)pjifRate))
goto laiFileErr;
break;
case FOURCC_seq:
/*
* If we find a seq chunk, read it into its preallocated
* space.
*/
picur = (DWORD *)((PBYTE)phcur + cpcur * sizeof(HCURSOR) +
cicur * sizeof(JIF));
if(!ReadChunk(pfi, &tag, (PBYTE)picur))
goto laiFileErr;
break;
case FOURCC_LIST:
{
DWORD cbChunk = (tag.ckSize + 1) & ~1;
/*
* See if this list is the 'fram' list of icon chunks
*/
if(!ReadFilePtrCopy(pfi, &tag.ckID, sizeof(tag.ckID))) {
goto laiFileErr;
}
cbChunk -= sizeof(tag.ckID);
if (tag.ckID != FOURCC_fram) {
/*
* Not the fram list (probably the INFO list). Skip
* the rest of this chunk. (Don't forget that we have
* already skipped one dword!)
*/
tag.ckSize = cbChunk;
SkipChunk(pfi, &tag);
break;
}
while(cbChunk >= sizeof(tag)) {
if (!ReadTag(pfi, &tag))
goto laiFileErr;
cbChunk -= sizeof(tag);
if(tag.ckID == FOURCC_icon) {
/*
* Ok, load the icon/cursor bits, create a cursor from
* them, and save a pointer to it away in the ACON
* cursor pointer array.
*/
phcur[ipcur] = ReadIconFromFileMap(pfi,
tag.ckSize,
cxDesired,
cyDesired,
LR_flags);
if (phcur[ipcur] == NULL) {
for (i = 0; i < ipcur; i++)
NtUserDestroyCursor(phcur[i], 0);
goto laiFileErr;
}
ipcur++;
} else {
/*
* Unknown chunk in fram list, just ignore it
*/
SkipChunk(pfi, &tag);
}
cbChunk -= (tag.ckSize + 1) & ~1;
}
}
break;
default:
/*
* We're not interested in this chunk, skip it.
*/
if(!SkipChunk(pfi, &tag))
goto laiFileErr;
break;
}
}
/*
* Sanity check the count of frames so we won't fault trying
* to select a nonexistant cursor
*/
if (cpcur != ipcur) {
RIPMSG2(RIP_WARNING, "LoadAniIcon: Invalid number of frames; Actual:%#lx Expected:%#lx",
ipcur, cpcur);
for (i = 0; i < ipcur; i++)
NtUserDestroyCursor(phcur[i], CURSOR_ALWAYSDESTROY);
goto laiFileErr;
}
if (cpcur != 0)
hacon = CreateAniIcon(pfi->pszName,
rt,
cicur,
picur,
cpcur,
phcur,
jifRate,
pjifRate,
LR_flags & LR_GLOBAL);
laiFileErr:
#if DBG
if (hacon == NULL) {
RIPMSG0(RIP_WARNING, "LoadAniIcon: Invalid icon data format");
}
#endif
if (panih != NULL)
UserLocalFree(panih);
return hacon;
}