Leaked source code of windows server 2003
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

683 lines
20 KiB

  1. /*==========================================================================
  2. *
  3. * Copyright (C) 1994-1998 Microsoft Corporation. All Rights Reserved.
  4. *
  5. * File: ddheapr.c
  6. * Content: Rectangular heap manager
  7. * History:
  8. * Date By Reason
  9. * ==== == ======
  10. * 30-mar-95 kylej initial implementation
  11. * 07-apr-95 kylej Added rectVidMemAmountFree
  12. * 15-may-95 craige made separate VMEM struct for rect & linear
  13. * 18-jun-95 craige specific pitch
  14. * 02-jul-95 craige have rectVidMemInit return a BOOL
  15. * 28-nov-95 colinmc new function to return amount of allocated memory
  16. * in a heap
  17. * 05-jul-96 colinmc Work Item: Removing the restriction on taking Win16
  18. * lock on VRAM surfaces (not including the primary)
  19. * 18-jan-97 colinmc Work Item: AGP support
  20. * 03-mar-97 jeffno Work item: Extended surface memory alignment
  21. * 03-Feb-98 DrewB Made portable between user and kernel.
  22. *
  23. ***************************************************************************/
  24. #include "ddrawpr.h"
  25. /****************************************************************************
  26. This memory manager manages allocation of rectangular blocks of
  27. video memory. It has essentially the same interface as the linear
  28. video memory manager implemented in vmemmgr.c. Memory allocations
  29. are tracked by nodes on two circular, doubly-linked lists; the free
  30. list and the alloc list. Each list has a special node called the
  31. sentinel which contains a special memory size. The head of each
  32. list always points to the sentinel node and the first member of the
  33. list (if there is one) is pointed to by the sentinel node. Block
  34. adjacency information is kept in each node so that several free nodes
  35. can be coalesced into larger free nodes. This takes place every
  36. time a block of memory is freed.
  37. This memory manager is designed to have no impact on video memory usage.
  38. Global memory is used to maintain the allocation and free lists. Because
  39. of this choice, merging of free blocks is a more expensive operation.
  40. The assumption is that in general, the speed of creating/destroying these
  41. memory blocks is not a high usage item and so it is OK to be slower.
  42. ****************************************************************************/
  43. /*
  44. * IS_FREE and NOT_FREE are used to set the free flag in the flags
  45. * field of each VMEM node. The free flag is the lsb of this field.
  46. */
  47. #define IS_FREE 0x00000001
  48. #define NOT_FREE 0xfffffffe
  49. /*
  50. * SENTINEL is the value stuffed into the size field of a VMEM
  51. * node to identify it as the sentinel node. This value makes
  52. * the assumption that no rectangle sized 0x7fff by 0xffff will
  53. * ever be requested.
  54. */
  55. #define SENTINEL 0x7fffffff
  56. /*
  57. * MIN_DIMENSION_SIZE determines the smallest valid dimension for a
  58. * free memory block. If dividing a rectangle will result in a
  59. * rectangle with a dimension less then MIN_DIMENSION_SIZE, the
  60. * rectangle is not divided.
  61. */
  62. #define MIN_DIMENSION_SIZE 4
  63. /*
  64. * BLOCK_BOUNDARY must be a power of 2, and at least 4. This gives
  65. * us the alignment of memory blocks.
  66. */
  67. #define BLOCK_BOUNDARY 4
  68. // This macro results in the free list being maintained with a
  69. // cx-major, cy-minor sort:
  70. #define CXCY(cx, cy) (((cx) << 16) | (cy))
  71. /*
  72. * Debugging helpers
  73. */
  74. #define DPFVMEMR(str,p) VDPF((0,V,"%s: %d,%d (%dx%d) ptr:%08x, block:%08x",str,p->x,p->y,p->cx,p->cy,p->ptr,p))
  75. #define CHECK_HEAP(a,b) ;
  76. /*
  77. * insertIntoDoubleList - add an item to the a list. The list is
  78. * kept in order of increasing size and is doubly linked. The
  79. * list is circular with a sentinel node indicating the end
  80. * of the list. The sentinel node has its size field set
  81. * to SENTINEL.
  82. */
  83. void insertIntoDoubleList( LPVMEMR pnew, LPVMEMR listhead )
  84. {
  85. LPVMEMR pvmem = listhead;
  86. #ifdef DEBUG
  87. if( pnew->size == 0 )
  88. {
  89. VDPF(( 0, V, "block size = 0!!!\n" ));
  90. }
  91. #endif
  92. /*
  93. * run through the list (sorted from smallest to largest) looking
  94. * for the first item bigger than the new item. If the sentinel
  95. * is encountered, insert the new item just before the sentinel.
  96. */
  97. while( pvmem->size != SENTINEL )
  98. {
  99. if( pnew->size < pvmem->size )
  100. {
  101. break;
  102. }
  103. pvmem = pvmem->next;
  104. }
  105. // insert the new item before the found one.
  106. pnew->prev = pvmem->prev;
  107. pnew->next = pvmem;
  108. pvmem->prev->next = pnew;
  109. pvmem->prev = pnew;
  110. } /* insertIntoDoubleList */
  111. /*
  112. * rectVidMemInit - initialize rectangular video memory manager
  113. */
  114. BOOL rectVidMemInit(
  115. LPVMEMHEAP pvmh,
  116. FLATPTR start,
  117. DWORD width,
  118. DWORD height,
  119. DWORD pitch )
  120. {
  121. LPVMEMR newNode;
  122. VDPF(( 4, V, "rectVidMemInit(start=%08lx,width=%ld,height=%ld,pitch=%ld)", start, width, height, pitch));
  123. pvmh->dwTotalSize = pitch * height;
  124. // Store the pitch for future address calculations.
  125. pvmh->stride = pitch;
  126. // Set up the Free list and the Alloc list by inserting the sentinel.
  127. pvmh->freeList = MemAlloc( sizeof(VMEMR) );
  128. if( pvmh->freeList == NULL )
  129. {
  130. return FALSE;
  131. }
  132. ((LPVMEMR)pvmh->freeList)->size = SENTINEL;
  133. ((LPVMEMR)pvmh->freeList)->cx = SENTINEL;
  134. ((LPVMEMR)pvmh->freeList)->cy = SENTINEL;
  135. ((LPVMEMR)pvmh->freeList)->next = (LPVMEMR)pvmh->freeList;
  136. ((LPVMEMR)pvmh->freeList)->prev = (LPVMEMR)pvmh->freeList;
  137. ((LPVMEMR)pvmh->freeList)->pLeft = NULL;
  138. ((LPVMEMR)pvmh->freeList)->pUp = NULL;
  139. ((LPVMEMR)pvmh->freeList)->pRight = NULL;
  140. ((LPVMEMR)pvmh->freeList)->pDown = NULL;
  141. pvmh->allocList = MemAlloc( sizeof(VMEMR) );
  142. if( pvmh->allocList == NULL )
  143. {
  144. MemFree(pvmh->freeList);
  145. return FALSE;
  146. }
  147. ((LPVMEMR)pvmh->allocList)->size = SENTINEL;
  148. ((LPVMEMR)pvmh->allocList)->next = (LPVMEMR)pvmh->allocList;
  149. ((LPVMEMR)pvmh->allocList)->prev = (LPVMEMR)pvmh->allocList;
  150. // Initialize the free list with the whole chunk of memory
  151. newNode = (LPVMEMR)MemAlloc( sizeof( VMEMR ) );
  152. if( newNode == NULL )
  153. {
  154. MemFree(pvmh->freeList);
  155. MemFree(pvmh->allocList);
  156. return FALSE;
  157. }
  158. newNode->ptr = start;
  159. newNode->size = CXCY(width, height);
  160. newNode->x = 0;
  161. newNode->y = 0;
  162. newNode->cx = width;
  163. newNode->cy = height;
  164. newNode->flags |= IS_FREE;
  165. newNode->pLeft = (LPVMEMR)pvmh->freeList;
  166. newNode->pUp = (LPVMEMR)pvmh->freeList;
  167. newNode->pRight = (LPVMEMR)pvmh->freeList;
  168. newNode->pDown = (LPVMEMR)pvmh->freeList;
  169. insertIntoDoubleList( newNode, ((LPVMEMR) pvmh->freeList)->next );
  170. return TRUE;
  171. } /* rectVidMemInit */
  172. /*
  173. * rectVidMemFini - done with rectangular video memory manager
  174. */
  175. void rectVidMemFini( LPVMEMHEAP pvmh )
  176. {
  177. LPVMEMR curr;
  178. LPVMEMR next;
  179. if( pvmh != NULL )
  180. {
  181. // free all memory allocated for the free list
  182. curr = ((LPVMEMR)pvmh->freeList)->next;
  183. while( curr->size != SENTINEL )
  184. {
  185. next = curr->next;
  186. MemFree( curr );
  187. curr = next;
  188. }
  189. MemFree( curr );
  190. pvmh->freeList = NULL;
  191. // free all memory allocated for the allocation list
  192. curr = ((LPVMEMR)pvmh->allocList)->next;
  193. while( curr->size != SENTINEL )
  194. {
  195. next = curr->next;
  196. MemFree( curr );
  197. curr = next;
  198. }
  199. MemFree( curr );
  200. pvmh->allocList = NULL;
  201. // free the heap data
  202. MemFree( pvmh );
  203. }
  204. } /* rectVidMemFini */
  205. /*
  206. * GetBeforeWastage.
  207. * Align the surface in the given block. Return the size of the holes
  208. * on the left side of the surface.
  209. * Fail if alignment would cause surface to spill out of block.
  210. * Works for horizontal and vertical alignment.
  211. * IN: dwBlockSize , dwBlockStart: Parameters of the block in which
  212. * the surface hopes to fit
  213. * dwSurfaceSize Width or height of the surface
  214. * dwAlignment Expected alignment. 0 means don't care
  215. * OUT: pdwBeforeWastage
  216. */
  217. BOOL GetBeforeWastage(
  218. DWORD dwBlockSize,
  219. DWORD dwBlockStart,
  220. DWORD dwSurfaceSize,
  221. LPDWORD pdwBeforeWastage,
  222. DWORD dwAlignment )
  223. {
  224. if (!dwAlignment)
  225. {
  226. *pdwBeforeWastage=0;
  227. /*
  228. * If no alignment requirement, then check if the surface fits
  229. */
  230. if (dwBlockSize >= dwSurfaceSize)
  231. {
  232. return TRUE;
  233. }
  234. return FALSE;
  235. }
  236. /*
  237. * There's an alignment.
  238. */
  239. *pdwBeforeWastage = (dwAlignment - (dwBlockStart % dwAlignment)) % dwAlignment;
  240. if ( *pdwBeforeWastage + dwSurfaceSize > dwBlockSize )
  241. {
  242. return FALSE;
  243. }
  244. DDASSERT( (dwBlockStart + *pdwBeforeWastage) % dwAlignment == 0 );
  245. return TRUE;
  246. }
  247. /*
  248. * rectVidMemAlloc - alloc some rectangular flat video memory
  249. */
  250. FLATPTR rectVidMemAlloc( LPVMEMHEAP pvmh, DWORD cxThis, DWORD cyThis,
  251. LPDWORD lpdwSize, LPSURFACEALIGNMENT lpAlignment )
  252. {
  253. LPVMEMR pvmem;
  254. DWORD cyRem;
  255. DWORD cxRem;
  256. DWORD cxBelow;
  257. DWORD cyBelow;
  258. DWORD cxBeside;
  259. DWORD cyBeside;
  260. LPVMEMR pnewBeside;
  261. LPVMEMR pnewBelow;
  262. DWORD dwXAlignment=0;
  263. DWORD dwYAlignment=0;
  264. DWORD dwLeftWastage=0;
  265. DWORD dwTopWastage=0;
  266. if((cxThis == 0) || (cyThis == 0) || (pvmh == NULL))
  267. return (FLATPTR) NULL;
  268. // Make sure the size of the block is a multiple of BLOCK_BOUNDARY
  269. // If every block allocated has a width which is a multiple of
  270. // BLOCK_BOUNDARY, it guarantees that all blocks will be allocated
  271. // on block boundaries.
  272. /*
  273. * Bump to new alignment
  274. */
  275. if( (cxThis >= (SENTINEL>>16) ) || (cyThis >= (SENTINEL&0xffff) ) )
  276. return (FLATPTR) NULL;
  277. if (lpAlignment)
  278. {
  279. dwXAlignment = lpAlignment->Rectangular.dwXAlignment;
  280. dwYAlignment = lpAlignment->Rectangular.dwYAlignment;
  281. }
  282. if (dwXAlignment < 4)
  283. {
  284. dwXAlignment = 4;
  285. }
  286. cxThis = (cxThis+(BLOCK_BOUNDARY-1)) & ~(BLOCK_BOUNDARY-1);
  287. /*
  288. * run through free list, looking for the closest matching block
  289. */
  290. pvmem = ((LPVMEMR)pvmh->freeList)->next;
  291. while (pvmem->size != SENTINEL)
  292. {
  293. if (!GetBeforeWastage( pvmem->cx, pvmem->x, cxThis, &dwLeftWastage, dwXAlignment ))
  294. {
  295. pvmem = pvmem->next;
  296. continue; //X size or alignment makes surface spill out of block
  297. }
  298. // Now see if size/alignment works for Y
  299. if (!GetBeforeWastage( pvmem->cy, pvmem->y, cyThis, &dwTopWastage, dwYAlignment ))
  300. {
  301. pvmem = pvmem->next;
  302. continue; //Y size alignment makes surface spill out of block
  303. }
  304. //success:
  305. break;
  306. }
  307. if(pvmem->size == SENTINEL)
  308. {
  309. // There was no rectangle large enough
  310. return (FLATPTR) NULL;
  311. }
  312. // pvmem now points to a rectangle that is the same size or larger
  313. // than the requested rectangle. We're going to use the upper-left
  314. // corner of the found rectangle and divide the unused remainder into
  315. // two rectangles which will go on the available list.
  316. // grow allocation by the wastage which makes the top-left aligned
  317. cxThis += dwLeftWastage;
  318. cyThis += dwTopWastage;
  319. // Compute the width of the unused rectangle to the right and the
  320. // height of the unused rectangle below:
  321. cyRem = pvmem->cy - cyThis;
  322. cxRem = pvmem->cx - cxThis;
  323. // Given finite area, we wish to find the two rectangles that are
  324. // most square -- i.e., the arrangement that gives two rectangles
  325. // with the least perimiter:
  326. cyBelow = cyRem;
  327. cxBeside = cxRem;
  328. if (cxRem <= cyRem)
  329. {
  330. cxBelow = cxThis + cxRem;
  331. cyBeside = cyThis;
  332. }
  333. else
  334. {
  335. cxBelow = cxThis;
  336. cyBeside = cyThis + cyRem;
  337. }
  338. // We only make new available rectangles of the unused right and
  339. // bottom portions if they're greater in dimension than MIN_DIMENSION_SIZE.
  340. // It hardly makes sense to do the book-work to keep around a
  341. // two pixel wide available space, for example.
  342. pnewBeside = NULL;
  343. if (cxBeside >= MIN_DIMENSION_SIZE)
  344. {
  345. pnewBeside = (LPVMEMR)MemAlloc( sizeof(VMEMR) );
  346. if( pnewBeside == NULL)
  347. return (FLATPTR) NULL;
  348. // Update the adjacency information along with the other required
  349. // information in this new node and then insert it into the free
  350. // list which is sorted in ascending cxcy.
  351. // size information
  352. pnewBeside->size = CXCY(cxBeside, cyBeside);
  353. pnewBeside->x = pvmem->x + cxThis;
  354. pnewBeside->y = pvmem->y;
  355. pnewBeside->ptr = pvmem->ptr + cxThis;
  356. pnewBeside->cx = cxBeside;
  357. pnewBeside->cy = cyBeside;
  358. pnewBeside->flags |= IS_FREE;
  359. // adjacency information
  360. pnewBeside->pLeft = pvmem;
  361. pnewBeside->pUp = pvmem->pUp;
  362. pnewBeside->pRight = pvmem->pRight;
  363. pnewBeside->pDown = pvmem->pDown;
  364. insertIntoDoubleList( pnewBeside, ((LPVMEMR) pvmh->freeList)->next);
  365. // Modify the current node to reflect the changes we've made:
  366. pvmem->cx = cxThis;
  367. }
  368. pnewBelow = NULL;
  369. if (cyBelow >= MIN_DIMENSION_SIZE)
  370. {
  371. pnewBelow = (LPVMEMR) MemAlloc( sizeof(VMEMR) );
  372. if (pnewBelow == NULL)
  373. return (FLATPTR) NULL;
  374. // Update the adjacency information along with the other required
  375. // information in this new node and then insert it into the free
  376. // list which is sorted in ascending cxcy.
  377. // size information
  378. pnewBelow->size = CXCY(cxBelow, cyBelow);
  379. pnewBelow->x = pvmem->x;
  380. pnewBelow->y = pvmem->y + cyThis;
  381. pnewBelow->ptr = pvmem->ptr + cyThis*pvmh->stride;
  382. pnewBelow->cx = cxBelow;
  383. pnewBelow->cy = cyBelow;
  384. pnewBelow->flags |= IS_FREE;
  385. // adjacency information
  386. pnewBelow->pLeft = pvmem->pLeft;
  387. pnewBelow->pUp = pvmem;
  388. pnewBelow->pRight = pvmem->pRight;
  389. pnewBelow->pDown = pvmem->pDown;
  390. insertIntoDoubleList( pnewBelow, ((LPVMEMR) pvmh->freeList)->next );
  391. // Modify the current node to reflect the changes we've made:
  392. pvmem->cy = cyThis;
  393. }
  394. // Update adjacency information for the current node
  395. if(pnewBelow != NULL)
  396. {
  397. pvmem->pDown = pnewBelow;
  398. if((pnewBeside != NULL) && (cyBeside == pvmem->cy))
  399. pnewBeside->pDown = pnewBelow;
  400. }
  401. if(pnewBeside != NULL)
  402. {
  403. pvmem->pRight = pnewBeside;
  404. if ((pnewBelow != NULL) && (cxBelow == pvmem->cx))
  405. pnewBelow->pRight = pnewBeside;
  406. }
  407. // Remove this node from the available list
  408. pvmem->next->prev = pvmem->prev;
  409. pvmem->prev->next = pvmem->next;
  410. // set up new pointers (pBits is the value returned to the client, pvmem
  411. // points to the actual top-left of the block).
  412. pvmem->pBits = pvmem->ptr + dwLeftWastage + dwTopWastage*pvmh->stride;
  413. pvmem->flags &= NOT_FREE;
  414. pvmem->size = CXCY(pvmem->cx, pvmem->cy);
  415. // Now insert it into the alloc list.
  416. insertIntoDoubleList( pvmem, ((LPVMEMR) pvmh->allocList)->next );
  417. if( NULL != lpdwSize )
  418. {
  419. /*
  420. * Note this is the total number of bytes needed for this surface
  421. * including the stuff off the left and right hand sides due to
  422. * pitch not being equal to width. This is different from the
  423. * size computed above which is simply the number of bytes within
  424. * the boundary of the surface itself.
  425. *
  426. * The formula below calculates the number of bytes from the first
  427. * byte in the rectangular surface to the first byte after it
  428. * taking the pitch into account. Complex I know but it works.
  429. */
  430. DDASSERT( 0UL != pvmem->cy );
  431. *lpdwSize = (pvmh->stride * (pvmem->cy - 1)) + pvmem->cx;
  432. }
  433. CHECK_HEAP("After rectVidMemAlloc",pvmh);
  434. return pvmem->pBits;
  435. } /* rectVidMemAlloc */
  436. /*
  437. * rectVidMemFree = free some rectangular flat video memory
  438. */
  439. void rectVidMemFree( LPVMEMHEAP pvmh, FLATPTR ptr )
  440. {
  441. LPVMEMR pvmem;
  442. LPVMEMR pBeside;
  443. // Find the node in the allocated list which matches ptr
  444. for(pvmem=((LPVMEMR)pvmh->allocList)->next; pvmem->size != SENTINEL;
  445. pvmem = pvmem->next)
  446. if(pvmem->pBits == ptr)
  447. break;
  448. if(pvmem->size == SENTINEL) // couldn't find allocated rectangle?
  449. {
  450. VDPF(( 0, V, "Couldn't find node requested freed!\n"));
  451. return;
  452. }
  453. // pvmem now points to the node which must be freed. Attempt to
  454. // coalesce rectangles around this node until no more action
  455. // is possible.
  456. while(1)
  457. {
  458. // Try merging with the right sibling:
  459. pBeside = pvmem->pRight;
  460. if ((pBeside->flags & IS_FREE) &&
  461. (pBeside->cy == pvmem->cy) &&
  462. (pBeside->pUp == pvmem->pUp) &&
  463. (pBeside->pDown == pvmem->pDown) &&
  464. (pBeside->pRight->pLeft != pBeside))
  465. {
  466. // Add the right rectangle to ours:
  467. pvmem->cx += pBeside->cx;
  468. pvmem->pRight = pBeside->pRight;
  469. // Remove pBeside from the list and free it.
  470. pBeside->next->prev = pBeside->prev;
  471. pBeside->prev->next = pBeside->next;
  472. MemFree(pBeside);
  473. continue; // go back and try again
  474. }
  475. // Try merging with the lower sibling:
  476. pBeside = pvmem->pDown;
  477. if ((pBeside->flags & IS_FREE) &&
  478. (pBeside->cx == pvmem->cx) &&
  479. (pBeside->pLeft == pvmem->pLeft) &&
  480. (pBeside->pRight == pvmem->pRight) &&
  481. (pBeside->pDown->pUp != pBeside))
  482. {
  483. pvmem->cy += pBeside->cy;
  484. pvmem->pDown = pBeside->pDown;
  485. // Remove pBeside from the list and free it.
  486. pBeside->next->prev = pBeside->prev;
  487. pBeside->prev->next = pBeside->next;
  488. MemFree(pBeside);
  489. continue; // go back and try again
  490. }
  491. // Try merging with the left sibling:
  492. pBeside = pvmem->pLeft;
  493. if ((pBeside->flags & IS_FREE) &&
  494. (pBeside->cy == pvmem->cy) &&
  495. (pBeside->pUp == pvmem->pUp) &&
  496. (pBeside->pDown == pvmem->pDown) &&
  497. (pBeside->pRight == pvmem) &&
  498. (pvmem->pRight->pLeft != pvmem))
  499. {
  500. // We add our rectangle to the one to the left:
  501. pBeside->cx += pvmem->cx;
  502. pBeside->pRight = pvmem->pRight;
  503. // Remove 'pvmem' from the list and free it:
  504. pvmem->next->prev = pvmem->prev;
  505. pvmem->prev->next = pvmem->next;
  506. MemFree(pvmem);
  507. pvmem = pBeside;
  508. continue;
  509. }
  510. // Try merging with the upper sibling:
  511. pBeside = pvmem->pUp;
  512. if ((pBeside->flags & IS_FREE) &&
  513. (pBeside->cx == pvmem->cx) &&
  514. (pBeside->pLeft == pvmem->pLeft) &&
  515. (pBeside->pRight == pvmem->pRight) &&
  516. (pBeside->pDown == pvmem) &&
  517. (pvmem->pDown->pUp != pvmem))
  518. {
  519. pBeside->cy += pvmem->cy;
  520. pBeside->pDown = pvmem->pDown;
  521. // Remove 'pvmem' from the list and free it:
  522. pvmem->next->prev = pvmem->prev;
  523. pvmem->prev->next = pvmem->next;
  524. MemFree(pvmem);
  525. pvmem = pBeside;
  526. continue;
  527. }
  528. // Remove the node from its current list.
  529. pvmem->next->prev = pvmem->prev;
  530. pvmem->prev->next = pvmem->next;
  531. pvmem->size = CXCY(pvmem->cx, pvmem->cy);
  532. pvmem->flags |= IS_FREE;
  533. // Insert the node into the free list:
  534. insertIntoDoubleList( pvmem, ((LPVMEMR) pvmh->freeList)->next );
  535. // No more area coalescing can be done, return.
  536. CHECK_HEAP("After rectVidMemFree",pvmh);
  537. return;
  538. }
  539. }
  540. /*
  541. * rectVidMemAmountAllocated
  542. */
  543. DWORD rectVidMemAmountAllocated( LPVMEMHEAP pvmh )
  544. {
  545. LPVMEMR pvmem;
  546. DWORD size;
  547. size = 0;
  548. // Traverse the alloc list and add up all the used space.
  549. for(pvmem=((LPVMEMR)pvmh->allocList)->next; pvmem->size != SENTINEL;
  550. pvmem = pvmem->next)
  551. {
  552. size += pvmem->cx * pvmem->cy;
  553. }
  554. return size;
  555. } /* rectVidMemAmountAllocated */
  556. /*
  557. * rectVidMemAmountFree
  558. */
  559. DWORD rectVidMemAmountFree( LPVMEMHEAP pvmh )
  560. {
  561. LPVMEMR pvmem;
  562. DWORD size;
  563. size = 0;
  564. // Traverse the free list and add up all the empty space.
  565. for(pvmem=((LPVMEMR)pvmh->freeList)->next; pvmem->size != SENTINEL;
  566. pvmem = pvmem->next)
  567. {
  568. size += pvmem->cx * pvmem->cy;
  569. }
  570. return size;
  571. } /* rectVidMemAmountFree */