Windows2003-3790/windows/core/dxkernel/dxg/ddheap.cxx
2020-09-30 16:53:55 +02:00

1345 lines
39 KiB
C++

/*==========================================================================
*
* Copyright (C) 1994-1999 Microsoft Corporation. All Rights Reserved.
*
* File: ddheap.c
* Content: Top-level heap routines.
* History:
* Date By Reason
* ==== == ======
* 06-dec-94 craige initial implementation
* 06-jan-95 craige integrated into DDRAW
* 20-mar-95 craige prepare for rectangular memory manager
* 27-mar-95 craige linear or rectangular vidmem
* 01-apr-95 craige happy fun joy updated header file
* 06-apr-95 craige fill in free video memory
* 15-may-95 craige made separate VMEM struct for rect & linear
* 10-jun-95 craige exported fns
* 02-jul-95 craige fail if VidMemInit if linear or rect. fail;
* removed linFindMemBlock
* 17-jul-95 craige added VidMemLargestFree
* 01-dec-95 colinmc added VidMemAmountAllocated
* 11-dec-95 kylej added VidMemGetRectStride
* 05-jul-96 colinmc Work Item: Removing the restriction on taking Win16
* lock on VRAM surfaces (not including the primary)
* 03-mar-97 jeffno Work item: Extended surface memory alignment
* 13-mar-97 colinmc Bug 6533: Pass uncached flag to VMM correctly
* 03-Feb-98 DrewB Made portable between user and kernel.
*
***************************************************************************/
#include "precomp.hxx"
/*
* VidMemInit - initialize video memory manager heap
*/
LPVMEMHEAP WINAPI VidMemInit(
DWORD flags,
FLATPTR start,
FLATPTR width_or_end,
DWORD height,
DWORD pitch )
{
LPVMEMHEAP pvmh;
pvmh = (LPVMEMHEAP)MemAlloc( sizeof( VMEMHEAP ) );
if( pvmh == NULL )
{
return NULL;
}
pvmh->dwFlags = flags;
ZeroMemory( & pvmh->Alignment.ddsCaps, sizeof(pvmh->Alignment.ddsCaps) );
if( pvmh->dwFlags & VMEMHEAP_LINEAR )
{
if( !linVidMemInit( pvmh, start, width_or_end ) )
{
MemFree( pvmh );
return NULL;
}
}
else
{
if( !rectVidMemInit( pvmh, start, (DWORD) width_or_end, height,
pitch ) )
{
MemFree( pvmh );
return NULL;
}
}
return pvmh;
} /* VidMemInit */
/*
* VidMemFini - done with video memory manager
*/
void WINAPI VidMemFini( LPVMEMHEAP pvmh )
{
if( pvmh->dwFlags & VMEMHEAP_LINEAR )
{
linVidMemFini( pvmh );
}
else
{
rectVidMemFini( pvmh );
}
} /* VidMemFini */
/*
* InternalVidMemAlloc - alloc some flat video memory and give us back the size
* we allocated
*/
FLATPTR WINAPI InternalVidMemAlloc( LPVMEMHEAP pvmh, DWORD x, DWORD y,
LPDWORD lpdwSize,
LPSURFACEALIGNMENT lpAlignment,
LPLONG lpNewPitch )
{
if( pvmh->dwFlags & VMEMHEAP_LINEAR )
{
return linVidMemAlloc( pvmh, x, y, lpdwSize, lpAlignment, lpNewPitch );
}
else
{
FLATPTR lp = rectVidMemAlloc( pvmh, x, y, lpdwSize, lpAlignment );
if (lp && lpNewPitch )
{
*lpNewPitch = (LONG) pvmh->stride;
}
return lp;
}
return (FLATPTR) NULL;
} /* InternalVidMemAlloc */
/*
* VidMemAlloc - alloc some flat video memory
*/
FLATPTR WINAPI VidMemAlloc( LPVMEMHEAP pvmh, DWORD x, DWORD y )
{
DWORD dwSize;
/*
* We are not interested in the size here.
*/
return InternalVidMemAlloc( pvmh, x, y, &dwSize , NULL , NULL );
} /* VidMemAlloc */
/*
* DxDdHeapVidMemFree = free some flat video memory
*/
void WINAPI DxDdHeapVidMemFree( LPVMEMHEAP pvmh, FLATPTR ptr )
{
if( pvmh->dwFlags & VMEMHEAP_LINEAR )
{
linVidMemFree( pvmh, ptr );
}
else
{
rectVidMemFree( pvmh, ptr );
}
} /* VidMemFree */
/*
* VidMemAmountAllocated
*/
DWORD WINAPI VidMemAmountAllocated( LPVMEMHEAP pvmh )
{
if( pvmh->dwFlags & VMEMHEAP_LINEAR )
{
return linVidMemAmountAllocated( pvmh );
}
else
{
return rectVidMemAmountAllocated( pvmh );
}
} /* VidMemAmountAllocated */
/*
* VidMemAmountFree
*/
DWORD WINAPI VidMemAmountFree( LPVMEMHEAP pvmh )
{
if( pvmh->dwFlags & VMEMHEAP_LINEAR )
{
return linVidMemAmountFree( pvmh );
}
else
{
return rectVidMemAmountFree( pvmh );
}
} /* VidMemAmountFree */
/*
* VidMemLargestFree
*/
DWORD WINAPI VidMemLargestFree( LPVMEMHEAP pvmh )
{
if( pvmh->dwFlags & VMEMHEAP_LINEAR )
{
return linVidMemLargestFree( pvmh );
}
else
{
return 0;
}
} /* VidMemLargestFree */
/*
* HeapVidMemInit
*
* Top level heap initialization code which handles AGP stuff.
*/
LPVMEMHEAP WINAPI HeapVidMemInit( LPVIDMEM lpVidMem,
DWORD pitch,
HANDLE hdev,
LPHEAPALIGNMENT pgad)
{
DWORD dwSize;
DDASSERT( NULL != lpVidMem );
if( lpVidMem->dwFlags & VIDMEM_ISNONLOCAL )
{
/*
* We do not actually call AGPReserve at this time since that would
* mean calling it each time a mode change occurred. Insted, we defer
* this until later when we call InitAgpHeap.
*/
/*
* Compute the size of the heap.
*/
if( lpVidMem->dwFlags & VIDMEM_ISLINEAR )
{
dwSize = (DWORD)(lpVidMem->fpEnd - lpVidMem->fpStart) + 1UL;
}
else
{
DDASSERT( lpVidMem->dwFlags & VIDMEM_ISRECTANGULAR );
dwSize = (pitch * lpVidMem->dwHeight);
}
DDASSERT( 0UL != dwSize );
/*
* Update the heap for the new start address
* (and end address for a linear heap).
*/
lpVidMem->fpStart = 0;
if( lpVidMem->dwFlags & VIDMEM_ISLINEAR )
{
lpVidMem->fpEnd = dwSize - 1UL;
}
else
{
DDASSERT( lpVidMem->dwFlags & VIDMEM_ISRECTANGULAR );
DDASSERT( pitch );
lpVidMem->dwHeight = dwSize / pitch;
}
}
if( lpVidMem->dwFlags & VIDMEM_ISLINEAR )
{
VDPF(( 1,V, "VidMemInit: Linear: fpStart = 0x%08x fpEnd = 0x%08x",
lpVidMem->fpStart, lpVidMem->fpEnd ));
lpVidMem->lpHeap = VidMemInit( VMEMHEAP_LINEAR, lpVidMem->fpStart,
lpVidMem->fpEnd, 0, 0 );
}
else
{
// We have no way of testing a rectangular AGP heap, so I'm disabling
// it for now.
if( !( lpVidMem->dwFlags & VIDMEM_ISNONLOCAL ) )
{
VDPF(( 1,V, "VidMemInit: Rectangular: fpStart = 0x%08x "
"dwWidth = %ld dwHeight = %ld, pitch = %ld",
lpVidMem->fpStart, lpVidMem->dwWidth, lpVidMem->dwHeight,
pitch ));
lpVidMem->lpHeap = VidMemInit( VMEMHEAP_RECTANGULAR, lpVidMem->fpStart,
lpVidMem->dwWidth, lpVidMem->dwHeight,
pitch );
}
}
/*
* Modify the caps and alt-caps so that you don't allocate local
* video memory surfaces out of AGP memory and vice-verse.
*/
if( lpVidMem->dwFlags & VIDMEM_ISNONLOCAL )
{
/*
* Its an AGP heap. So don't let explict LOCAL video memory
* be allocated out of this heap.
*/
lpVidMem->ddsCaps.dwCaps |= DDSCAPS_LOCALVIDMEM;
lpVidMem->ddsCapsAlt.dwCaps |= DDSCAPS_LOCALVIDMEM;
}
else
{
/*
* Its a local video memory heap. So don't let explicity NON-LOCAL
* video memory be allocated out of this heap.
*/
lpVidMem->ddsCaps.dwCaps |= DDSCAPS_NONLOCALVIDMEM;
lpVidMem->ddsCapsAlt.dwCaps |= DDSCAPS_NONLOCALVIDMEM;
}
/*
* Copy any extended alignment data into the private heap structure
*/
if ( lpVidMem->lpHeap )
{
if ( pgad )
{
lpVidMem->lpHeap->dwFlags |= VMEMHEAP_ALIGNMENT;
lpVidMem->lpHeap->Alignment = *pgad;
VDPF((5,V,"Extended alignment turned on for this heap."));
VDPF((6,V,"Alignments are turned on for:"));
VDPF((6,V," %08X",pgad->ddsCaps));
}
else
{
/*
* This means the allocation routines will do no alignment modifications
*/
VDPF((5,V,"Extended alignment turned OFF for this heap."));
lpVidMem->lpHeap->dwFlags &= ~VMEMHEAP_ALIGNMENT;
}
}
return lpVidMem->lpHeap;
} /* HeapVidMemInit */
/*
* HeapVidMemFini
*
* Top level heap release code. Handle AGP stuff
*/
void WINAPI HeapVidMemFini( LPVIDMEM lpVidMem, HANDLE hdev )
{
DWORD dwCommittedSize = 0UL;
PVOID pvReservation;
BYTE* pAgpCommitMask = NULL;
DWORD dwAgpCommitMaskSize;
DWORD dwTotalSize;
/*
* Remember how much memory we committed to the AGP heap.
*/
DDASSERT( NULL != lpVidMem->lpHeap );
if( lpVidMem->dwFlags & VIDMEM_ISNONLOCAL )
{
dwCommittedSize = lpVidMem->lpHeap->dwCommitedSize;
pvReservation = lpVidMem->lpHeap->pvPhysRsrv;
pAgpCommitMask = lpVidMem->lpHeap->pAgpCommitMask;
dwAgpCommitMaskSize = lpVidMem->lpHeap->dwAgpCommitMaskSize;
dwTotalSize = lpVidMem->lpHeap->dwTotalSize;
}
/*
* Free the memory manager
*/
VidMemFini( lpVidMem->lpHeap );
lpVidMem->lpHeap = NULL;
if( lpVidMem->dwFlags & VIDMEM_ISNONLOCAL )
{
BOOL fSuccess = TRUE;
/*
* If this is a non-local (AGP) heap then decommit and
* free the GART memory now.
*/
if( ( 0UL != dwCommittedSize ) &&
( pAgpCommitMask != NULL ) )
{
DWORD dwTemp;
/*
* Only decommit if we actually bothered to commit something
* in the first place.
*/
fSuccess = AGPDecommitAll( hdev, pvReservation,
pAgpCommitMask,
dwAgpCommitMaskSize,
&dwTemp,
dwTotalSize);
/*
* Should never fail and not much we can do if it does apart
* from assert that something bad is happening.
*/
DDASSERT( fSuccess );
}
if( pAgpCommitMask != NULL )
{
VFREEMEM(pAgpCommitMask);
}
if( pvReservation != NULL )
{
fSuccess = AGPFree( hdev, pvReservation );
}
/*
* Again this should only fail if the OS is in an unstable state
* or if I have screwed up (sadly the later is all too likely)
* so assert.
*/
DDASSERT( fSuccess );
}
} /* HeapVidMemFini */
/*
* This is an external entry point which can be used by drivers to allocate
* aligned surfaces.
*/
FLATPTR WINAPI DxDdHeapVidMemAllocAligned(
LPVIDMEM lpVidMem,
DWORD dwWidth,
DWORD dwHeight,
LPSURFACEALIGNMENT lpAlignment ,
LPLONG lpNewPitch )
{
HANDLE hdev;
FLATPTR ptr;
DWORD dwSize;
if ( lpVidMem == NULL ||
lpVidMem->lpHeap == NULL ||
(lpVidMem->dwFlags & VIDMEM_HEAPDISABLED))
{
return (FLATPTR) NULL;
}
if( lpVidMem->dwFlags & VIDMEM_ISNONLOCAL )
{
if( lpVidMem->lpHeap->pvPhysRsrv == NULL )
{
LPVIDMEM pHeap;
DWORD dwHeap;
// If we haven't yet initialized the AGP heap, then we will
// do so now. This could be dangerous since initializing the
// heap causes the driver to get re-entered by the
// UpdateNonLocalVidMemHeap call.
EDD_DIRECTDRAW_GLOBAL* peDirectDrawGlobal =
(EDD_DIRECTDRAW_GLOBAL*) OsGetAGPDeviceHandle(lpVidMem->lpHeap);
pHeap = peDirectDrawGlobal->pvmList;
for (dwHeap = 0;
dwHeap < peDirectDrawGlobal->dwNumHeaps;
pHeap++, dwHeap++)
{
if( pHeap == lpVidMem )
{
break;
}
}
if( dwHeap < peDirectDrawGlobal->dwNumHeaps )
{
InitAgpHeap( peDirectDrawGlobal,
dwHeap,
(HANDLE) peDirectDrawGlobal );
}
if( ( lpVidMem->lpHeap->pvPhysRsrv == NULL ) ||
( lpVidMem->dwFlags & VIDMEM_HEAPDISABLED ) )
{
return (FLATPTR) NULL;
}
}
/*
* As we may need to commit AGP memory we need a device handle
* to communicate with the AGP controller. Rather than hunting
* through the driver object list hoping we will find a
* local object for this process we just create a handle
* and discard it after the allocation. This should not be
* performance critical code to start with.
*/
hdev = OsGetAGPDeviceHandle(lpVidMem->lpHeap);
if (hdev == NULL)
{
return 0;
}
}
else
{
hdev = NULL;
}
/* Pass NULL Alignment and new pitch pointer */
ptr = HeapVidMemAlloc( lpVidMem, dwWidth, dwHeight,
hdev, lpAlignment, lpNewPitch, &dwSize );
if( hdev != NULL )
{
OsCloseAGPDeviceHandle( hdev );
}
return ptr;
}
/*
* HeapVidMemAlloc
*
* Top level video memory allocation function. Handles AGP stuff
*/
FLATPTR WINAPI HeapVidMemAlloc( LPVIDMEM lpVidMem, DWORD x, DWORD y,
HANDLE hdev, LPSURFACEALIGNMENT lpAlignment,
LPLONG lpNewPitch, LPDWORD pdwSize )
{
FLATPTR fpMem;
DWORD dwSize;
DDASSERT( NULL != lpVidMem );
DDASSERT( NULL != lpVidMem->lpHeap );
/*
* Validate the reserved and flags fields of the alignment structure
*/
if( lpAlignment )
{
if( ( lpAlignment->Linear.dwReserved2 != 0 ) ||
( lpAlignment->Linear.dwFlags & ~( SURFACEALIGN_DISCARDABLE ) ) )
{
return NULL;
}
}
if( ( lpVidMem->dwFlags & VIDMEM_ISNONLOCAL ) &&
( lpVidMem->lpHeap->pvPhysRsrv == NULL ) )
{
return NULL;
}
fpMem = InternalVidMemAlloc( lpVidMem->lpHeap, x, y, &dwSize,
lpAlignment, lpNewPitch );
if( 0UL == fpMem )
{
return fpMem;
}
if( lpVidMem->dwFlags & VIDMEM_ISNONLOCAL )
{
DWORD dwCommittedSize;
BOOL fSuccess;
DWORD dwOffset;
dwOffset = (DWORD)(fpMem - lpVidMem->fpStart);
/*
* Okay, we have the offset and the size we need to commit. So ask
* the OS to commit memory to that portion of this previously
* reserved GART range.
*/
fSuccess = AGPCommit( hdev, lpVidMem->lpHeap->pvPhysRsrv,
dwOffset, dwSize,
lpVidMem->lpHeap->pAgpCommitMask,
&dwCommittedSize,
lpVidMem->lpHeap->dwTotalSize );
if( !fSuccess )
{
/*
* Couldn't commit. Must be out of memory.
* Put the allocated memory back and fail.
*/
DxDdHeapVidMemFree( lpVidMem->lpHeap, fpMem );
return (FLATPTR) NULL;
}
lpVidMem->lpHeap->dwCommitedSize += dwCommittedSize;
/*
* Now we need to vitually commit the memory for all of the
* DirectDrawLocals that we have. This is because some drivers
* (nvidia) allocate a single buffer and then hand out pointers
* into that buffer to multimple processes.
*/
EDD_DIRECTDRAW_GLOBAL* peDirectDrawGlobal =
(EDD_DIRECTDRAW_GLOBAL*) OsGetAGPDeviceHandle(lpVidMem->lpHeap);
EDD_DIRECTDRAW_LOCAL* peDirectDrawLocal;
LPVIDMEM pHeap;
DWORD dwHeap;
pHeap = peDirectDrawGlobal->pvmList;
for (dwHeap = 0;
dwHeap < peDirectDrawGlobal->dwNumHeaps;
pHeap++, dwHeap++)
{
if( pHeap == lpVidMem )
{
break;
}
}
if( dwHeap < peDirectDrawGlobal->dwNumHeaps )
{
peDirectDrawLocal = peDirectDrawGlobal->peDirectDrawLocalList;
while( ( peDirectDrawLocal != NULL ) && fSuccess )
{
if( !( peDirectDrawLocal->fl & DD_LOCAL_DISABLED ) )
{
fSuccess = AGPCommitVirtual( peDirectDrawLocal,
lpVidMem,
dwHeap,
dwOffset,
dwSize);
}
peDirectDrawLocal = peDirectDrawLocal->peDirectDrawLocalNext;
}
}
else
{
fSuccess = FALSE;
}
if( !fSuccess )
{
/*
* Something went wrong on the virtual commit, so fail the allocation.
*/
DxDdHeapVidMemFree( lpVidMem->lpHeap, fpMem );
return (FLATPTR) NULL;
}
}
if (pdwSize != NULL)
{
*pdwSize = dwSize;
}
return fpMem;
} /* HeapVidMemAlloc */
/*
* IsDifferentPixelFormat
*
* determine if two pixel formats are the same or not
*
* (CMcC) 12/14/95 Really useful - so no longer static
*
* This function really shouldn't be in a heap file but it's
* needed by both the user and kernel code so this is a convenient
* place to put it to have it shared.
*/
BOOL IsDifferentPixelFormat( LPDDPIXELFORMAT pdpf1, LPDDPIXELFORMAT pdpf2 )
{
/*
* same flags?
*/
if( pdpf1->dwFlags != pdpf2->dwFlags )
{
VDPF(( 8, S, "Flags differ!" ));
return TRUE;
}
/*
* same bitcount for non-YUV surfaces?
*/
if( !(pdpf1->dwFlags & (DDPF_YUV | DDPF_FOURCC)) )
{
if( pdpf1->dwRGBBitCount != pdpf2->dwRGBBitCount )
{
VDPF(( 8, S, "RGB Bitcount differs!" ));
return TRUE;
}
}
/*
* same RGB properties?
*/
if( pdpf1->dwFlags & DDPF_RGB )
{
if( pdpf1->dwRBitMask != pdpf2->dwRBitMask )
{
VDPF(( 8, S, "RBitMask differs!" ));
return TRUE;
}
if( pdpf1->dwGBitMask != pdpf2->dwGBitMask )
{
VDPF(( 8, S, "GBitMask differs!" ));
return TRUE;
}
if( pdpf1->dwBBitMask != pdpf2->dwBBitMask )
{
VDPF(( 8, S, "BBitMask differs!" ));
return TRUE;
}
if( pdpf1->dwRGBAlphaBitMask != pdpf2->dwRGBAlphaBitMask )
{
VDPF(( 8, S, "RGBAlphaBitMask differs!" ));
return TRUE;
}
}
/*
* same YUV properties?
*/
if( pdpf1->dwFlags & DDPF_YUV )
{
VDPF(( 8, S, "YUV???" ));
if( pdpf1->dwFourCC != pdpf2->dwFourCC )
{
return TRUE;
}
if( pdpf1->dwYUVBitCount != pdpf2->dwYUVBitCount )
{
return TRUE;
}
if( pdpf1->dwYBitMask != pdpf2->dwYBitMask )
{
return TRUE;
}
if( pdpf1->dwUBitMask != pdpf2->dwUBitMask )
{
return TRUE;
}
if( pdpf1->dwVBitMask != pdpf2->dwVBitMask )
{
return TRUE;
}
if( pdpf1->dwYUVAlphaBitMask != pdpf2->dwYUVAlphaBitMask )
{
return TRUE;
}
}
/*
* Possible to use FOURCCs w/o setting the DDPF_YUV flag
* ScottM 7/11/96
*/
else if( pdpf1->dwFlags & DDPF_FOURCC )
{
VDPF(( 8, S, "FOURCC???" ));
if( pdpf1->dwFourCC != pdpf2->dwFourCC )
{
return TRUE;
}
}
/*
* If Interleaved Z then check Z bit masks are the same
*/
if( pdpf1->dwFlags & DDPF_ZPIXELS )
{
VDPF(( 8, S, "ZPIXELS???" ));
if( pdpf1->dwRGBZBitMask != pdpf2->dwRGBZBitMask )
return TRUE;
}
return FALSE;
} /* IsDifferentPixelFormat */
/*
* SurfaceCapsToAlignment
*
* Return a pointer to the appropriate alignment element in a VMEMHEAP
* structure given surface caps.
*
*/
LPSURFACEALIGNMENT SurfaceCapsToAlignment(
LPVIDMEM lpVidmem ,
LPDDRAWI_DDRAWSURFACE_LCL lpSurfaceLcl,
LPVIDMEMINFO lpVidMemInfo)
{
LPVMEMHEAP lpHeap;
LPDDSCAPS lpCaps;
LPDDRAWI_DDRAWSURFACE_GBL lpSurfaceGbl;
DDASSERT( lpVidmem );
DDASSERT( lpSurfaceLcl );
DDASSERT( lpVidMemInfo );
DDASSERT( lpVidmem->lpHeap );
if ( !lpVidmem->lpHeap )
return NULL;
lpCaps = &lpSurfaceLcl->ddsCaps;
lpHeap = lpVidmem->lpHeap;
lpSurfaceGbl = lpSurfaceLcl->lpGbl;
if ( (lpHeap->dwFlags & VMEMHEAP_ALIGNMENT) == 0 )
return NULL;
if ( lpCaps->dwCaps & DDSCAPS_EXECUTEBUFFER )
{
if ( lpHeap->Alignment.ddsCaps.dwCaps & DDSCAPS_EXECUTEBUFFER )
{
VDPF((6,V,"Aligning surface as execute buffer"));
return & lpHeap->Alignment.ExecuteBuffer;
}
/*
* If the surface is an execute buffer, then no other
* alignment can apply
*/
return NULL;
}
if ( lpCaps->dwCaps & DDSCAPS_OVERLAY )
{
if ( lpHeap->Alignment.ddsCaps.dwCaps & DDSCAPS_OVERLAY )
{
VDPF((6,V,"Aligning surface as overlay"));
return & lpHeap->Alignment.Overlay;
}
/*
* If the surface is an overlay, then no other alignment can apply
*/
return NULL;
}
if ( lpCaps->dwCaps & DDSCAPS_TEXTURE )
{
if ( lpHeap->Alignment.ddsCaps.dwCaps & DDSCAPS_TEXTURE )
{
VDPF((6,V,"Aligning surface as texture"));
return & lpHeap->Alignment.Texture;
}
/*
* If it's a texture, it can't be an offscreen or any of the others
*/
return NULL;
}
if ( lpCaps->dwCaps & DDSCAPS_ZBUFFER )
{
if ( lpHeap->Alignment.ddsCaps.dwCaps & DDSCAPS_ZBUFFER )
{
VDPF((6,V,"Aligning surface as Z buffer"));
return & lpHeap->Alignment.ZBuffer;
}
return NULL;
}
if ( lpCaps->dwCaps & DDSCAPS_ALPHA )
{
if ( lpHeap->Alignment.ddsCaps.dwCaps & DDSCAPS_ALPHA )
{
VDPF((6,V,"Aligning surface as alpha buffer"));
return & lpHeap->Alignment.AlphaBuffer;
}
return NULL;
}
/*
* We need to give a surface which may potentially become a back buffer
* the alignment which is reserved for potentially visible back buffers.
* This includes any surface which has made it through the above checks
* and has the same dimensions as the primary.
* Note we check only the dimensions of the primary. There's an outside
* chance that an app could create its back buffer before it creates
* the primary
*/
do
{
if ( lpSurfaceLcl->dwFlags & DDRAWISURF_HASPIXELFORMAT )
{
if (IsDifferentPixelFormat( &lpVidMemInfo->ddpfDisplay,
&lpSurfaceGbl->ddpfSurface ))
{
/*
* Different pixel format from primary means this surface
* cannot be part of primary chain
*/
break;
}
}
if ( (DWORD)lpSurfaceGbl->wWidth != lpVidMemInfo->dwDisplayWidth )
break;
if ( (DWORD)lpSurfaceGbl->wHeight != lpVidMemInfo->dwDisplayHeight )
break;
/*
* This surface could potentially be part of primary chain.
* It has the same
* pixel format as the primary and the same dimensions.
*/
if ( lpHeap->Alignment.ddsCaps.dwCaps & DDSCAPS_FLIP )
{
VDPF((6,V,"Aligning surface as potential primary surface"));
return & lpHeap->Alignment.FlipTarget;
}
/*
* Drop through and check for offscreen if driver specified no
* part-of-primary-chain alignment
*/
break;
} while (0);
if ( lpCaps->dwCaps & DDSCAPS_OFFSCREENPLAIN )
{
if ( lpHeap->Alignment.ddsCaps.dwCaps & DDSCAPS_OFFSCREENPLAIN )
{
VDPF((6,V,"Aligning surface as offscreen plain"));
return & lpHeap->Alignment.Offscreen;
}
}
VDPF((6,V,"No extended alignment for surface"));
return NULL;
}
/*
* DdHeapAlloc
*
* Search all heaps for one that has space and the appropriate
* caps for the requested surface type and size.
*
* We AND the caps bits required and the caps bits not allowed
* by the video memory. If the result is zero, it is OK.
*
* This is called in 2 passes. Pass1 is the preferred memory state,
* pass2 is the "oh no no memory" state.
*
* On pass1, we use ddsCaps in the VIDMEM struct.
* On pass2, we use ddsCapsAlt in the VIDMEM struct.
*
*/
FLATPTR DdHeapAlloc( DWORD dwNumHeaps,
LPVIDMEM pvmHeaps,
HANDLE hdev,
LPVIDMEMINFO lpVidMemInfo,
DWORD dwWidth,
DWORD dwHeight,
LPDDRAWI_DDRAWSURFACE_LCL lpSurfaceLcl,
DWORD dwFlags,
LPVIDMEM *ppvmHeap,
LPLONG plNewPitch,
LPDWORD pdwNewCaps,
LPDWORD pdwSize)
{
LPVIDMEM pvm;
DWORD vm_caps;
int i;
FLATPTR pvidmem;
HANDLE hvxd;
LPDDSCAPS lpCaps;
LPDDSCAPSEX lpExtendedRestrictions;
LPDDSCAPSEX lpExtendedCaps;
DDASSERT( NULL != pdwNewCaps );
DDASSERT( NULL != lpSurfaceLcl );
lpCaps = &lpSurfaceLcl->ddsCaps;
lpExtendedCaps = &lpSurfaceLcl->lpSurfMore->ddsCapsEx;
for( i = 0 ; i < (int)dwNumHeaps ; i++ )
{
pvm = &pvmHeaps[i];
// Skip disabled heaps.
if (pvm->dwFlags & VIDMEM_HEAPDISABLED)
{
continue;
}
/*
* Skip rectangular heaps if we were told to.
*/
if (dwFlags & DDHA_SKIPRECTANGULARHEAPS)
{
if (pvm->dwFlags & VIDMEM_ISRECTANGULAR)
{
continue;
}
}
/*
* If local or non-local video memory has been explicity
* specified then ignore heaps which don't match the required
* memory type.
*/
if( ( lpCaps->dwCaps & DDSCAPS_LOCALVIDMEM ) &&
( pvm->dwFlags & VIDMEM_ISNONLOCAL ) )
{
VDPF(( 4, V, "Local video memory was requested but heap is "
"non local. Ignoring heap %d", i ));
continue;
}
if( ( lpCaps->dwCaps & DDSCAPS_NONLOCALVIDMEM ) &&
!( pvm->dwFlags & VIDMEM_ISNONLOCAL ) )
{
VDPF(( 4, V, "Non-local video memory was requested but "
"heap is local. Ignoring heap %d", i ));
continue;
}
if( !( lpCaps->dwCaps & DDSCAPS_NONLOCALVIDMEM ) &&
( pvm->dwFlags & VIDMEM_ISNONLOCAL ) &&
( dwFlags & DDHA_ALLOWNONLOCALMEMORY ) )
{
/*
* We can allow textures to fail over to DMA model cards
* if the card exposes an appropriate heap. This won't
* affect cards which can't texture from nonlocal, because
* they won't expose such a heap. This mod doesn't affect
* execute model because all surfaces fail over to nonlocal
* for them.
* Note that we should only fail over to nonlocal if the
* surface wasn't explicitly requested in local. There is a
* clause a few lines up which guarantees this.
*/
if ( !(lpCaps->dwCaps & DDSCAPS_TEXTURE) )
{
VDPF(( 4, V, "Non-local memory not explicitly requested "
"for non-texture surface. Ignoring non-local heap %d",
i ));
continue;
}
/*
* If the device can't texture out of AGP, we need to fail this
* heap, since the app is probably expecting to texture out of
* this surface.
*/
if ( !(dwFlags & DDHA_ALLOWNONLOCALTEXTURES) )
{
continue;
}
}
if( dwFlags & DDHA_USEALTCAPS )
{
vm_caps = pvm->ddsCapsAlt.dwCaps;
lpExtendedRestrictions = &(pvm->lpHeap->ddsCapsExAlt);
}
else
{
vm_caps = pvm->ddsCaps.dwCaps;
lpExtendedRestrictions = &(pvm->lpHeap->ddsCapsEx);
}
if( ((lpCaps->dwCaps & vm_caps) == 0) &&
((lpExtendedRestrictions->dwCaps2 & lpExtendedCaps->dwCaps2) == 0) &&
((lpExtendedRestrictions->dwCaps3 & lpExtendedCaps->dwCaps3) == 0) &&
((lpExtendedRestrictions->dwCaps4 & lpExtendedCaps->dwCaps4) == 0))
{
pvidmem = HeapVidMemAlloc(
pvm,
dwWidth,
dwHeight,
hdev,
SurfaceCapsToAlignment(pvm, lpSurfaceLcl, lpVidMemInfo),
plNewPitch,
pdwSize);
if( pvidmem != (FLATPTR) NULL )
{
*ppvmHeap = pvm;
if( pvm->dwFlags & VIDMEM_ISNONLOCAL )
*pdwNewCaps |= DDSCAPS_NONLOCALVIDMEM;
else
*pdwNewCaps |= DDSCAPS_LOCALVIDMEM;
return pvidmem;
}
}
}
return (FLATPTR) NULL;
} /* DdHeapAlloc */
/*
* GetHeapSizeInPages
*
* This is called to determine how much memory should be allocated
* for the commit mask. Initially the mask had an entry for each page,
* so it returned the number of pages in the heap, but the commit mask now has
* a bit for each 16 page chunk, so we now return the number of chunks in the
* heap.
*/
DWORD GetHeapSizeInPages(LPVIDMEM lpVidMem, LONG pitch)
{
DWORD dwSize;
if( lpVidMem->dwFlags & VIDMEM_ISLINEAR )
{
dwSize = (DWORD)(lpVidMem->fpEnd - lpVidMem->fpStart) + 1UL;
}
else
{
DDASSERT( lpVidMem->dwFlags & VIDMEM_ISRECTANGULAR );
dwSize = (pitch * lpVidMem->dwHeight);
}
return AGPGetChunkCount(dwSize);
}
/*
* CleanupAgpCommits
*
* Some drivers leave outstanding allocates after an app exits so that we
* cannot completely cleanup the AGP heap, so what this function does is
* decommit as much of the physical AGP memory as it can. To do this, it
* determines what memory is still allocated and then decommits everything
* esle.
*/
VOID CleanupAgpCommits(LPVIDMEM lpVidMem, HANDLE hdev, EDD_DIRECTDRAW_GLOBAL* peDirectDrawGlobal, int iHeapIndex)
{
BYTE* pGoodCommits;
BYTE* pAgpCommitMask;
BYTE* pVirtCommitMask;
BYTE* pTempMask;
DWORD dwAgpCommitMaskSize;
DWORD i;
DWORD dwOffset;
DWORD dwSize;
DWORD dwDecommittedSize;
EDD_DIRECTDRAW_LOCAL* peDirectDrawLocal;
EDD_VMEMMAPPING* peMap;
if( ( lpVidMem->lpHeap == NULL ) ||
( lpVidMem->lpHeap->pAgpCommitMask == NULL ) ||
( lpVidMem->lpHeap->dwAgpCommitMaskSize == 0 ) ||
( lpVidMem->lpHeap->dwCommitedSize == 0 ) ||
( lpVidMem->lpHeap->pvPhysRsrv == NULL ) )
{
return;
}
pAgpCommitMask = lpVidMem->lpHeap->pAgpCommitMask;
dwAgpCommitMaskSize = lpVidMem->lpHeap->dwAgpCommitMaskSize;
pGoodCommits = (BYTE*) PALLOCMEM(dwAgpCommitMaskSize,
'pddG');
if( pGoodCommits == NULL )
{
return;
}
pVirtCommitMask = (BYTE*) PALLOCMEM(dwAgpCommitMaskSize,
'pddG');
if( pVirtCommitMask == NULL )
{
VFREEMEM(pGoodCommits);
return;
}
/*
* Walk the alloc list and build the list of all the pages that should
* be committed.
*/
if( lpVidMem->lpHeap->dwFlags & VMEMHEAP_LINEAR )
{
LPVMEML pAlloc = (LPVMEML) lpVidMem->lpHeap->allocList;
while( pAlloc != NULL )
{
dwOffset = (DWORD)(pAlloc->ptr - lpVidMem->fpStart);
dwSize = pAlloc->size;
AGPUpdateCommitMask( pGoodCommits, dwOffset, dwSize, lpVidMem->lpHeap->dwTotalSize );
pAlloc = pAlloc->next;
}
}
else
{
LPVMEMR pAlloc = (LPVMEMR) lpVidMem->lpHeap->allocList;
/*
* This is a circular list where the end of the list is denoted by
* a node containing a sentinel value of 0x7fffffff
*/
while(( pAlloc != NULL ) &&
( pAlloc->size != 0x7fffffff ))
{
dwOffset = (DWORD)(pAlloc->ptr - lpVidMem->fpStart);
dwSize = (lpVidMem->lpHeap->stride *
(pAlloc->cy - 1)) + pAlloc->cx;
AGPUpdateCommitMask( pGoodCommits, dwOffset, dwSize, lpVidMem->lpHeap->dwTotalSize );
pAlloc = pAlloc->next;
}
}
/*
* Check here to verify that every page that we think should be committed
* actually is committed.
*/
#if DBG
{
BYTE bit;
DWORD dwPage;
DWORD dwNumPages;
bit = 1;
dwNumPages = dwAgpCommitMaskSize * BITS_IN_BYTE;
dwPage = 0;
while( dwPage < dwNumPages )
{
ASSERTGDI(!((!(pAgpCommitMask[dwPage/BITS_IN_BYTE] & bit)&&(pGoodCommits[dwPage/BITS_IN_BYTE] & bit))),
"Page not commited when we think it should be!");
if( bit == 0x80 )
{
bit = 1;
}
else
{
bit <<= 1;
}
dwPage++;
}
}
#endif
/*
* Now build a list of pages that are committed but that don't need to be.
* To save space, we will re-use pAgpCommitMask for this purpose.
*/
for( i = 0; i < dwAgpCommitMaskSize; i++ )
{
pAgpCommitMask[i] ^= pGoodCommits[i];
}
/*
* We don't want to physically decommit the memory w/o first virtually
* decommitting it for each processs.
*/
peDirectDrawLocal = peDirectDrawGlobal->peDirectDrawLocalList;
while( peDirectDrawLocal != NULL )
{
if (peDirectDrawLocal->ppeMapAgp != NULL)
{
peMap = peDirectDrawLocal->ppeMapAgp[iHeapIndex];
if ((peMap != NULL) && !(peDirectDrawLocal->fl & DD_LOCAL_DISABLED))
{
// Replace the committed mask with the mask that we want to decommit
ASSERTGDI((dwAgpCommitMaskSize == peMap->dwAgpVirtualCommitMaskSize) ,
"Virtual AGP mask size does not equal physical mask size");
memcpy( pVirtCommitMask, pAgpCommitMask, dwAgpCommitMaskSize );
pTempMask = peMap->pAgpVirtualCommitMask;
peMap->pAgpVirtualCommitMask = pVirtCommitMask;
// pVirtCommitMask now contains all of the pages that are
// physically committed but that don't need to be, but that
// doesn't mean that these are virtually commited, so we AND
// out the pages that are not virtually committed.
for( i = 0; i < dwAgpCommitMaskSize; i++ )
{
pVirtCommitMask[i] &= pTempMask[i];
}
AGPDecommitVirtual( peMap,
peDirectDrawLocal->peDirectDrawGlobal,
peDirectDrawLocal,
lpVidMem->lpHeap->dwTotalSize);
// pTempMask contains all of the pages that used to be virtually
// committed but are not neccessarily committed now. pGoodCommits
// contains all of the physical pages that need to kept, so ANDing
// them will give us the pages that are currently committed.
peMap->pAgpVirtualCommitMask = pTempMask;
for( i = 0; i < dwAgpCommitMaskSize; i++ )
{
pTempMask[i] &= pGoodCommits[i];
}
}
}
peDirectDrawLocal = peDirectDrawLocal->peDirectDrawLocalNext;
}
AGPDecommitAll( hdev,
lpVidMem->lpHeap->pvPhysRsrv,
pAgpCommitMask,
dwAgpCommitMaskSize,
&dwDecommittedSize,
lpVidMem->lpHeap->dwTotalSize);
lpVidMem->lpHeap->dwCommitedSize -= dwDecommittedSize;
memcpy( pAgpCommitMask, pGoodCommits, dwAgpCommitMaskSize );
VFREEMEM(pGoodCommits);
VFREEMEM(pVirtCommitMask);
}
/*
* SwapHeaps
*
* During a mode change, we create a new heap and copy it over the old heap.
* For AGP heaps, however, we really only want one instance of the heap
* active at any given time, so the new heap is not fully initialized. This
* means that we just swapped our good heap with a abd one, which means that
* we need to selectively swap certain elements of the heap back.
*/
void SwapHeaps( LPVIDMEM pOldVidMem, LPVIDMEM pNewVidMem )
{
LPVMEMHEAP pOldHeap = pOldVidMem->lpHeap;
LPVMEMHEAP pNewHeap = pNewVidMem->lpHeap;
FLATPTR fpTemp;
DWORD dwTemp;
LPVOID pvTemp;
LARGE_INTEGER liTemp;
fpTemp = pOldVidMem->fpStart;
pOldVidMem->fpStart = pNewVidMem->fpStart;
pNewVidMem->fpStart = fpTemp;
fpTemp = pOldVidMem->fpEnd;
pOldVidMem->fpEnd = pNewVidMem->fpEnd;
pNewVidMem->fpEnd = fpTemp;
dwTemp = pOldHeap->stride;
pOldHeap->stride = pNewHeap->stride;
pNewHeap->stride = dwTemp;
pvTemp = pOldHeap->freeList;
pOldHeap->freeList = pNewHeap->freeList;
pNewHeap->freeList = pvTemp;
pvTemp = pOldHeap->allocList;
pOldHeap->allocList = pNewHeap->allocList;
pNewHeap->allocList = pvTemp;
pOldHeap->dwTotalSize = min(pOldHeap->dwTotalSize,pNewHeap->dwTotalSize);
fpTemp = pOldHeap->fpGARTLin;
pOldHeap->fpGARTLin = pNewHeap->fpGARTLin;
pNewHeap->fpGARTLin = fpTemp;
fpTemp = pOldHeap->fpGARTDev;
pOldHeap->fpGARTDev = pNewHeap->fpGARTDev;
pNewHeap->fpGARTDev = fpTemp;
dwTemp = pOldHeap->dwCommitedSize;
pOldHeap->dwCommitedSize = pNewHeap->dwCommitedSize;
pNewHeap->dwCommitedSize = dwTemp;
liTemp = pOldHeap->liPhysAGPBase;
pOldHeap->liPhysAGPBase = pNewHeap->liPhysAGPBase;
pNewHeap->liPhysAGPBase = liTemp;
pvTemp = pOldHeap->pvPhysRsrv;
pOldHeap->pvPhysRsrv = pNewHeap->pvPhysRsrv;
pNewHeap->pvPhysRsrv = pvTemp;
pvTemp = (LPVOID) pOldHeap->pAgpCommitMask;
pOldHeap->pAgpCommitMask = pNewHeap->pAgpCommitMask;
pNewHeap->pAgpCommitMask = (BYTE*) pvTemp;
dwTemp = pOldHeap->dwAgpCommitMaskSize;
pOldHeap->dwAgpCommitMaskSize = pNewHeap->dwAgpCommitMaskSize;
pNewHeap->dwAgpCommitMaskSize = dwTemp;
}