xref: /reactos/dll/win32/kernel32/client/heapmem.c (revision 5100859e)
1 /*
2  * PROJECT:         ReactOS Win32 Base API
3  * LICENSE:         GPL - See COPYING in the top level directory
4  * FILE:            dll/win32/kernel32/client/heapmem.c
5  * PURPOSE:         Heap Memory APIs (wrappers for RtlHeap*)
6  * PROGRAMMERS:     Alex Ionescu (alex.ionescu@reactos.org)
7  */
8 
9 /* INCLUDES *******************************************************************/
10 
11 #include <k32.h>
12 
13 #define NDEBUG
14 #include <debug.h>
15 
16 /* GLOBALS ********************************************************************/
17 
18 RTL_HANDLE_TABLE BaseHeapHandleTable;
19 HANDLE BaseHeap;
20 ULONG_PTR SystemRangeStart;
21 
22 /* PRIVATE FUNCTIONS **********************************************************/
23 
24 VOID
25 NTAPI
26 BaseDllInitializeMemoryManager(VOID)
27 {
28     BaseHeap = RtlGetProcessHeap();
29     RtlInitializeHandleTable(0xFFFF,
30                              sizeof(BASE_HEAP_HANDLE_ENTRY),
31                              &BaseHeapHandleTable);
32     NtQuerySystemInformation(SystemRangeStartInformation,
33                              &SystemRangeStart,
34                              sizeof(SystemRangeStart),
35                              NULL);
36 }
37 
38 /* PUBLIC FUNCTIONS ***********************************************************/
39 
40 /*
41  * @implemented
42  */
43 HANDLE
44 WINAPI
45 HeapCreate(DWORD flOptions,
46            SIZE_T dwInitialSize,
47            SIZE_T dwMaximumSize)
48 {
49     HANDLE hRet;
50     ULONG Flags;
51 
52     /* Remove non-Win32 flags and tag this allocation */
53     Flags = (flOptions & (HEAP_GENERATE_EXCEPTIONS | HEAP_NO_SERIALIZE)) |
54             HEAP_CLASS_1;
55 
56     /* Check if heap is growable and ensure max size is correct */
57     if (dwMaximumSize == 0)
58         Flags |= HEAP_GROWABLE;
59     else if (dwMaximumSize < BaseStaticServerData->SysInfo.PageSize &&
60             dwInitialSize > dwMaximumSize)
61     {
62         /* Max size is non-zero but less than page size which can't be correct.
63            Fix it up by bumping it to the initial size whatever it is. */
64         dwMaximumSize = dwInitialSize;
65     }
66 
67     /* Call RTL Heap */
68     hRet = RtlCreateHeap(Flags,
69                          NULL,
70                          dwMaximumSize,
71                          dwInitialSize,
72                          NULL,
73                          NULL);
74 
75     /* Set the last error if we failed, and return the pointer */
76     if (!hRet) SetLastError(ERROR_NOT_ENOUGH_MEMORY);
77     return hRet;
78 }
79 
80 /*
81  * @implemented
82  */
83 BOOL
84 WINAPI
85 HeapDestroy(HANDLE hHeap)
86 {
87     /* Return TRUE if the heap was destroyed */
88    if (!RtlDestroyHeap(hHeap)) return TRUE;
89 
90     /* Otherwise, we got the handle back, so fail */
91     SetLastError(ERROR_INVALID_HANDLE);
92     return FALSE;
93 }
94 
95 /*
96  * @implemented
97  */
98 HANDLE
99 WINAPI
100 GetProcessHeap(VOID)
101 {
102     /* Call the RTL API */
103     return RtlGetProcessHeap();
104 }
105 
106 /*
107  * @implemented
108  */
109 DWORD
110 WINAPI
111 GetProcessHeaps(DWORD NumberOfHeaps,
112                 PHANDLE ProcessHeaps)
113 {
114     /* Call the RTL API */
115     return RtlGetProcessHeaps(NumberOfHeaps, ProcessHeaps);
116 }
117 
118 /*
119  * @implemented
120  */
121 BOOL
122 WINAPI
123 HeapLock(HANDLE hHeap)
124 {
125     /* Call the RTL API */
126     return RtlLockHeap(hHeap);
127 }
128 
129 /*
130  * @implemented
131  */
132 BOOL
133 WINAPI
134 HeapUnlock(HANDLE hHeap)
135 {
136     /* Call the RTL API */
137     return RtlUnlockHeap(hHeap);
138 }
139 
140 /*
141  * @implemented
142  */
143 SIZE_T
144 WINAPI
145 HeapCompact(HANDLE hHeap, DWORD dwFlags)
146 {
147     /* Call the RTL API */
148     return RtlCompactHeap(hHeap, dwFlags);
149 }
150 
151 /*
152  * @implemented
153  */
154 BOOL
155 WINAPI
156 HeapValidate(HANDLE hHeap,
157              DWORD dwFlags,
158              LPCVOID lpMem)
159 {
160     /* Call the RTL API */
161     return RtlValidateHeap(hHeap, dwFlags, (PVOID)lpMem);
162 }
163 
164 /*
165  * @implemented
166  */
167 DWORD
168 WINAPI
169 HeapCreateTagsW(HANDLE hHeap,
170                 DWORD dwFlags,
171                 PWSTR lpTagName,
172                 PWSTR lpTagSubName)
173 {
174     /* Call the RTL API */
175     return RtlCreateTagHeap(hHeap,
176                             dwFlags,
177                             lpTagName,
178                             lpTagSubName);
179 }
180 
181 /*
182  * @implemented
183  */
184 DWORD
185 WINAPI
186 HeapExtend(HANDLE hHeap,
187            DWORD dwFlags,
188            PVOID BaseAddress,
189            DWORD dwBytes)
190 {
191     NTSTATUS Status;
192 
193     /* Call the RTL API. Gone in Vista, so commented out. */
194     Status = STATUS_NOT_IMPLEMENTED; //RtlExtendHeap(hHeap, dwFlags, BaseAddress, dwBytes);
195     if (!NT_SUCCESS(Status))
196     {
197         /* We failed */
198         BaseSetLastNTError(Status);
199         return FALSE;
200     }
201 
202     /* Return success */
203     return TRUE;
204 }
205 
206 /*
207  * @implemented
208  */
209 PWSTR
210 WINAPI
211 HeapQueryTagW(HANDLE hHeap,
212               DWORD dwFlags,
213               WORD wTagIndex,
214               BOOL bResetCounters,
215               PVOID lpTagInfo)
216 {
217     /* Call the RTL API */
218     return RtlQueryTagHeap(hHeap,
219                            dwFlags,
220                            wTagIndex,
221                            (BOOLEAN)bResetCounters,
222                            lpTagInfo);
223 }
224 
225 /*
226  * @implemented
227  */
228 BOOL
229 WINAPI
230 HeapSummary(HANDLE hHeap,
231             DWORD dwFlags,
232             PVOID Summary)
233 {
234     NTSTATUS Status;
235     RTL_HEAP_USAGE Usage;
236 
237     /* Fill in the length information */
238     Usage.Length = sizeof(Usage);
239 
240     /* Call RTL. Gone in Vista, so commented out */
241     Status = STATUS_NOT_IMPLEMENTED; //RtlUsageHeap(hHeap, dwFlags, &Usage);
242     if (!NT_SUCCESS(Status))
243     {
244         /* We failed */
245         BaseSetLastNTError(Status);
246         return FALSE;
247     }
248 
249     /* FIXME: Summary == Usage?! */
250     RtlCopyMemory(Summary, &Usage, sizeof(Usage));
251     return TRUE;
252 }
253 
254 /*
255  * @implemented
256  */
257 BOOL
258 WINAPI
259 HeapUsage(HANDLE hHeap,
260           DWORD dwFlags,
261           DWORD Unknown,
262           DWORD Unknown2,
263           IN PVOID Usage)
264 {
265     NTSTATUS Status;
266 
267     /* Call RTL. Gone in Vista, so commented out */
268     Status = STATUS_NOT_IMPLEMENTED; //RtlUsageHeap(hHeap, dwFlags, &Usage);
269     if (!NT_SUCCESS(Status))
270     {
271         /* We failed */
272         BaseSetLastNTError(Status);
273         return FALSE;
274     }
275     else if (Status == STATUS_MORE_ENTRIES)
276     {
277         /* There are still more entries to parse */
278         return TRUE;
279     }
280 
281     /* Otherwise, we're completely done, so we return FALSE, but NO_ERROR */
282     SetLastError(NO_ERROR);
283     return FALSE;
284 }
285 
286 /*
287  * @implemented
288  */
289 BOOL
290 WINAPI
291 HeapWalk(HANDLE	hHeap,
292          LPPROCESS_HEAP_ENTRY lpEntry)
293 {
294     NTSTATUS Status;
295 
296     DPRINT1("Warning, HeapWalk is calling RtlWalkHeap with Win32 parameters\n");
297 
298     Status = RtlWalkHeap(hHeap, lpEntry);
299 
300     if (!NT_SUCCESS(Status))
301     {
302         SetLastError(RtlNtStatusToDosError(Status));
303         return FALSE;
304     }
305 
306     return TRUE;
307 }
308 
309 /*
310  * @implemented
311  */
312 BOOL
313 WINAPI
314 HeapQueryInformation(HANDLE HeapHandle,
315                      HEAP_INFORMATION_CLASS HeapInformationClass,
316                      PVOID HeapInformation OPTIONAL,
317                      SIZE_T HeapInformationLength OPTIONAL,
318                      PSIZE_T ReturnLength OPTIONAL)
319 {
320     NTSTATUS Status;
321 
322     Status = RtlQueryHeapInformation(HeapHandle,
323                                      HeapInformationClass,
324                                      HeapInformation,
325                                      HeapInformationLength,
326                                      ReturnLength);
327 
328     if (!NT_SUCCESS(Status))
329     {
330         BaseSetLastNTError(Status);
331         return FALSE;
332     }
333 
334     return TRUE;
335 }
336 
337 /*
338  * @implemented
339  */
340 BOOL
341 WINAPI
342 HeapSetInformation(HANDLE HeapHandle,
343                    HEAP_INFORMATION_CLASS HeapInformationClass,
344                    PVOID HeapInformation OPTIONAL,
345                    SIZE_T HeapInformationLength OPTIONAL)
346 {
347     NTSTATUS Status;
348 
349     Status = RtlSetHeapInformation(HeapHandle,
350                                    HeapInformationClass,
351                                    HeapInformation,
352                                    HeapInformationLength);
353 
354     if (!NT_SUCCESS(Status))
355     {
356         BaseSetLastNTError(Status);
357         return FALSE;
358     }
359 
360     return TRUE;
361 }
362 
363 /*
364  * @implemented
365  */
366 HGLOBAL
367 NTAPI
368 GlobalAlloc(UINT uFlags,
369             SIZE_T dwBytes)
370 {
371     ULONG Flags = 0;
372     PVOID Ptr = NULL;
373     HANDLE hMemory;
374     PBASE_HEAP_HANDLE_ENTRY HandleEntry;
375     BASE_TRACE_ALLOC(dwBytes, uFlags);
376     ASSERT(BaseHeap);
377 
378     /* Make sure the flags are valid */
379     if (uFlags & ~GMEM_VALID_FLAGS)
380     {
381         /* They aren't, fail */
382         BASE_TRACE_FAILURE();
383         SetLastError(ERROR_INVALID_PARAMETER);
384         return NULL;
385     }
386 
387     /* Convert ZEROINIT */
388     if (uFlags & GMEM_ZEROINIT) Flags |= HEAP_ZERO_MEMORY;
389 
390     /* Check if we're not movable, which means pointer-based heap */
391     if (!(uFlags & GMEM_MOVEABLE))
392     {
393         /* Check if this is DDESHARE (deprecated) */
394         if (uFlags & GMEM_DDESHARE) Flags |= BASE_HEAP_ENTRY_FLAG_DDESHARE;
395 
396         /* Allocate heap for it */
397         Ptr = RtlAllocateHeap(BaseHeap, Flags, dwBytes ? dwBytes : 1);
398         if (!Ptr) SetLastError(ERROR_NOT_ENOUGH_MEMORY);
399         BASE_TRACE_ALLOC2(Ptr);
400         return Ptr;
401     }
402 
403     /* This is heap based, so lock it in first */
404     RtlLockHeap(BaseHeap);
405 
406     /*
407      * Disable locking, enable custom flags, and write the
408      * movable flag (deprecated)
409      */
410     Flags |= HEAP_NO_SERIALIZE |
411              HEAP_SETTABLE_USER_VALUE |
412              BASE_HEAP_FLAG_MOVABLE;
413 
414     /* Allocate the handle */
415     HandleEntry = BaseHeapAllocEntry();
416     if (!HandleEntry)
417     {
418         /* Fail */
419         hMemory = NULL;
420         SetLastError(ERROR_NOT_ENOUGH_MEMORY);
421         BASE_TRACE_FAILURE();
422     }
423     else
424     {
425         /* Get the object and make sure we have size */
426         hMemory = &HandleEntry->Object;
427         if (dwBytes)
428         {
429             /* Allocate the actual memory for it */
430             Ptr = RtlAllocateHeap(BaseHeap, Flags, dwBytes);
431             BASE_TRACE_PTR(HandleEntry, Ptr);
432             if (!Ptr)
433             {
434                 /* We failed, manually set the allocate flag and free the handle */
435                 HandleEntry->Flags = RTL_HANDLE_VALID;
436                 BaseHeapFreeEntry(HandleEntry);
437 
438                 /* For the cleanup case */
439                 HandleEntry = NULL;
440             }
441             else
442             {
443                 /* All worked well, save our heap entry */
444                 RtlSetUserValueHeap(BaseHeap, HEAP_NO_SERIALIZE, Ptr, hMemory);
445             }
446         }
447     }
448 
449     /* Cleanup! First unlock the heap */
450     RtlUnlockHeap(BaseHeap);
451 
452     /* Check if a handle was allocated */
453     if (HandleEntry)
454     {
455         /* Set the pointer and allocated flag */
456         HandleEntry->Object = Ptr;
457         HandleEntry->Flags = RTL_HANDLE_VALID;
458         if (!Ptr)
459         {
460             /* We don't have a valid pointer, but so reuse this handle */
461             HandleEntry->Flags |= BASE_HEAP_ENTRY_FLAG_REUSE;
462         }
463 
464         /* Check if the handle is discardable */
465         if (uFlags & GMEM_DISCARDABLE)
466         {
467             /* Save it in the handle entry */
468             HandleEntry->Flags |= BASE_HEAP_ENTRY_FLAG_REUSABLE;
469         }
470 
471         /* Check if the handle is moveable */
472         if (uFlags & GMEM_MOVEABLE)
473         {
474             /* Save it in the handle entry */
475             HandleEntry->Flags |= BASE_HEAP_ENTRY_FLAG_MOVABLE;
476         }
477 
478         /* Check if the handle is DDE Shared */
479         if (uFlags & GMEM_DDESHARE)
480         {
481             /* Save it in the handle entry */
482             HandleEntry->Flags |= BASE_HEAP_ENTRY_FLAG_DDESHARE;
483         }
484 
485         /* Set the pointer */
486         Ptr = hMemory;
487     }
488 
489     /* Return the pointer */
490     return Ptr;
491 }
492 
493 /*
494  * @implemented
495  */
496 SIZE_T
497 NTAPI
498 GlobalCompact(DWORD dwMinFree)
499 {
500     /* Call the RTL Heap Manager */
501     return RtlCompactHeap(BaseHeap, 0);
502 }
503 
504 /*
505  * @implemented
506  */
507 VOID
508 NTAPI
509 GlobalFix(HGLOBAL hMem)
510 {
511     /* Lock the memory if it the handle is valid */
512     if (INVALID_HANDLE_VALUE != hMem) GlobalLock(hMem);
513 }
514 
515 /*
516  * @implemented
517  */
518 UINT
519 NTAPI
520 GlobalFlags(HGLOBAL hMem)
521 {
522     PBASE_HEAP_HANDLE_ENTRY HandleEntry;
523     HANDLE Handle = NULL;
524     ULONG Flags = 0;
525     UINT uFlags = GMEM_INVALID_HANDLE;
526 
527     /* Start by locking the heap */
528     RtlLockHeap(BaseHeap);
529     _SEH2_TRY
530     {
531         /* Check if this is a simple RTL Heap Managed block */
532         if (!((ULONG_PTR)hMem & BASE_HEAP_IS_HANDLE_ENTRY))
533         {
534             /* Then we'll query RTL Heap */
535             RtlGetUserInfoHeap(BaseHeap, Flags, hMem, &Handle, &Flags);
536             BASE_TRACE_PTR(Handle, hMem);
537 
538             /*
539              * Check if RTL Heap didn't find a handle associated with us or
540              * said that this heap isn't movable, which means something we're
541              * really not a handle-based heap.
542              */
543             if (!(Handle) || !(Flags & BASE_HEAP_FLAG_MOVABLE))
544             {
545                 /* Then set the flags to 0 */
546                 uFlags = 0;
547             }
548             else
549             {
550                 /* Otherwise we're handle-based, so get the internal handle */
551                 hMem = Handle;
552             }
553         }
554 
555         /* Check if the handle is actually an entry in our table */
556         if ((ULONG_PTR)hMem & BASE_HEAP_IS_HANDLE_ENTRY)
557         {
558             /* Then get the entry */
559             HandleEntry = BaseHeapGetEntry(hMem);
560             BASE_TRACE_HANDLE(HandleEntry, hMem);
561 
562             /* Make sure it's a valid handle */
563             if (BaseHeapValidateEntry(HandleEntry))
564             {
565                 /* Get the lock count first */
566                 uFlags = HandleEntry->LockCount & GMEM_LOCKCOUNT;
567 
568                 /* Now check if it's discardable */
569                 if (HandleEntry->Flags & BASE_HEAP_ENTRY_FLAG_REUSABLE)
570                 {
571                     /* Set the Win32 Flag */
572                     uFlags |= GMEM_DISCARDABLE;
573                 }
574 
575                 /* Check if it's DDE Shared */
576                 if (HandleEntry->Flags & BASE_HEAP_ENTRY_FLAG_DDESHARE)
577                 {
578                     /* Set the Win32 Flag */
579                     uFlags |= GMEM_DDESHARE;
580                 }
581 
582                 /* Now check if it's discarded */
583                 if (HandleEntry->Flags & BASE_HEAP_ENTRY_FLAG_REUSE)
584                 {
585                    /* Set the Win32 Flag */
586                    uFlags |= GMEM_DISCARDED;
587                }
588             }
589         }
590 
591         /* Check if by now, we still haven't gotten any useful flags */
592         if (uFlags == GMEM_INVALID_HANDLE) SetLastError(ERROR_INVALID_HANDLE);
593     }
594     _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
595     {
596         /* Set the exception code */
597         BaseSetLastNTError(_SEH2_GetExceptionCode());
598     }
599     _SEH2_END;
600 
601     /* All done! Unlock heap and return Win32 Flags */
602     RtlUnlockHeap(BaseHeap);
603     return uFlags;
604 }
605 
606 /*
607  * @implemented
608  */
609 HGLOBAL
610 NTAPI
611 GlobalFree(HGLOBAL hMem)
612 {
613     PBASE_HEAP_HANDLE_ENTRY HandleEntry;
614     LPVOID Ptr;
615     BASE_TRACE_DEALLOC(hMem);
616 
617     /* Check if this was a simple allocated heap entry */
618     if (!((ULONG_PTR)hMem & BASE_HEAP_IS_HANDLE_ENTRY))
619     {
620         /* Free it with the RTL Heap Manager */
621         if (RtlFreeHeap(BaseHeap, 0, hMem))
622         {
623             /* Return NULL since there's no handle */
624             return NULL;
625         }
626         else
627         {
628             /* Otherwise fail */
629             BASE_TRACE_FAILURE();
630             SetLastError(ERROR_INVALID_HANDLE);
631             return hMem;
632         }
633     }
634 
635     /* It's a handle probably, so lock the heap */
636     RtlLockHeap(BaseHeap);
637     _SEH2_TRY
638     {
639         /* Make sure that this is an entry in our handle database */
640         if ((ULONG_PTR)hMem & BASE_HEAP_IS_HANDLE_ENTRY)
641         {
642             /* Get the entry */
643             HandleEntry = BaseHeapGetEntry(hMem);
644             BASE_TRACE_HANDLE(HandleEntry, hMem);
645 
646             /* Make sure the handle is valid */
647             if (!BaseHeapValidateEntry(HandleEntry))
648             {
649                 /* It's not, fail */
650                 SetLastError(ERROR_INVALID_HANDLE);
651                 Ptr = NULL;
652             }
653             else
654             {
655                 /* It's valid, so get the pointer */
656                 Ptr = HandleEntry->Object;
657 
658                 /* Free this handle */
659                 BaseHeapFreeEntry(HandleEntry);
660 
661                 /* If the pointer is 0, then we don't have a handle either */
662                 if (!Ptr) hMem = NULL;
663             }
664         }
665         else
666         {
667             /* Otherwise, reuse the handle as a pointer */
668             BASE_TRACE_FAILURE();
669             Ptr = hMem;
670         }
671 
672         /* Check if we got here with a valid heap pointer */
673         if (Ptr)
674         {
675             /* Free it with the RTL Heap Manager */
676             if (RtlFreeHeap(BaseHeap, HEAP_NO_SERIALIZE, Ptr))
677             {
678                 /* Everything worked */
679                 hMem = NULL;
680             }
681             else
682             {
683                 /* This wasn't a real heap handle */
684                 SetLastError(ERROR_INVALID_HANDLE);
685             }
686         }
687     }
688     _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
689     {
690         /* Set the exception code */
691         BaseSetLastNTError(_SEH2_GetExceptionCode());
692     }
693     _SEH2_END;
694 
695     /* We're done, so unlock the heap and return the handle */
696     RtlUnlockHeap(BaseHeap);
697     return hMem;
698 }
699 
700 /*
701  * @implemented
702  */
703 HGLOBAL
704 NTAPI
705 GlobalHandle(LPCVOID pMem)
706 {
707     HANDLE Handle = NULL;
708     ULONG Flags;
709 
710     /* Lock the heap */
711     RtlLockHeap(BaseHeap);
712     _SEH2_TRY
713     {
714         /* Query RTL Heap */
715         if (!RtlGetUserInfoHeap(BaseHeap,
716                                 HEAP_NO_SERIALIZE,
717                                 (PVOID)pMem,
718                                 &Handle,
719                                 &Flags))
720         {
721             /* RTL Heap Manager does not know about this heap */
722             SetLastError(ERROR_INVALID_HANDLE);
723         }
724         else
725         {
726             /*
727              * Check if RTL Heap didn't find a handle for us or said that
728              * this heap isn't movable.
729              */
730             BASE_TRACE_PTR(Handle, pMem);
731             if (!(Handle) || !(Flags & BASE_HEAP_FLAG_MOVABLE))
732             {
733                 /* We're actually handle-based, so the pointer is a handle */
734                 Handle = (HANDLE)pMem;
735             }
736         }
737     }
738     _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
739     {
740         /* Set the exception code */
741         BaseSetLastNTError(_SEH2_GetExceptionCode());
742     }
743     _SEH2_END;
744 
745     /* All done, unlock the heap and return the handle */
746     RtlUnlockHeap(BaseHeap);
747     return Handle;
748 }
749 
750 /*
751  * @implemented
752  */
753 LPVOID
754 NTAPI
755 GlobalLock(HGLOBAL hMem)
756 {
757     PBASE_HEAP_HANDLE_ENTRY HandleEntry;
758     LPVOID Ptr;
759 
760     /* Check if this was a simple allocated heap entry */
761     if (!((ULONG_PTR)hMem & BASE_HEAP_IS_HANDLE_ENTRY))
762     {
763         /* Make sure it's not a kernel or invalid address */
764         if ((hMem >= (HGLOBAL)SystemRangeStart) || (IsBadReadPtr(hMem, 1)))
765         {
766             /* Signal an error */
767             SetLastError(ERROR_INVALID_HANDLE);
768             return NULL;
769         }
770 
771         /* It's all good */
772         return hMem;
773     }
774 
775     /* Otherwise, lock the heap */
776     RtlLockHeap(BaseHeap);
777     _SEH2_TRY
778     {
779         /* Get the handle entry */
780         HandleEntry = BaseHeapGetEntry(hMem);
781         BASE_TRACE_HANDLE(HandleEntry, hMem);
782 
783         /* Make sure it's valid */
784         if (!BaseHeapValidateEntry(HandleEntry))
785         {
786             /* It's not, fail */
787             BASE_TRACE_FAILURE();
788             SetLastError(ERROR_INVALID_HANDLE);
789             Ptr = NULL;
790         }
791         else
792         {
793             /* Otherwise, get the pointer */
794             Ptr = HandleEntry->Object;
795             if (Ptr)
796             {
797                 /* Increase the lock count, unless we've went too far */
798                 if (HandleEntry->LockCount++ == GMEM_LOCKCOUNT)
799                 {
800                     /* In which case we simply unlock once */
801                     HandleEntry->LockCount--;
802                 }
803             }
804             else
805             {
806                 /* The handle is still there but the memory was already freed */
807                 SetLastError(ERROR_DISCARDED);
808             }
809         }
810     }
811     _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
812     {
813         SetLastError(ERROR_INVALID_HANDLE);
814         Ptr = NULL;
815     }
816     _SEH2_END;
817 
818     /* All done. Unlock the heap and return the pointer */
819     RtlUnlockHeap(BaseHeap);
820     return Ptr;
821 }
822 
823 HGLOBAL
824 NTAPI
825 GlobalReAlloc(HGLOBAL hMem,
826               SIZE_T dwBytes,
827               UINT uFlags)
828 {
829     PBASE_HEAP_HANDLE_ENTRY HandleEntry;
830     HANDLE Handle;
831     LPVOID Ptr;
832     ULONG Flags = 0;
833 
834     /* Throw out invalid flags */
835     if (uFlags & ~(GMEM_VALID_FLAGS | GMEM_MODIFY))
836     {
837         SetLastError(ERROR_INVALID_PARAMETER);
838         return NULL;
839     }
840 
841     /* Throw out invalid combo */
842     if ((uFlags & GMEM_DISCARDABLE) && !(uFlags & GMEM_MODIFY))
843     {
844         SetLastError(ERROR_INVALID_PARAMETER);
845         return NULL;
846     }
847 
848     /* Convert ZEROINIT */
849     if (uFlags & GMEM_ZEROINIT) Flags |= HEAP_ZERO_MEMORY;
850 
851     /* If this wasn't a movable heap, then we MUST re-alloc in place */
852     if (!(uFlags & GMEM_MOVEABLE)) Flags |= HEAP_REALLOC_IN_PLACE_ONLY;
853 
854     /* Lock the heap and disable built-in locking in the RTL Heap functions */
855     RtlLockHeap(BaseHeap);
856     Flags |= HEAP_NO_SERIALIZE;
857 
858     /* Check if this is a simple handle-based block */
859     if (((ULONG_PTR)hMem & BASE_HEAP_IS_HANDLE_ENTRY))
860     {
861         /* Get the entry */
862         HandleEntry = BaseHeapGetEntry(hMem);
863         BASE_TRACE_HANDLE(HandleEntry, hMem);
864 
865         /* Make sure the handle is valid */
866         if (!BaseHeapValidateEntry(HandleEntry))
867         {
868             /* Fail */
869             BASE_TRACE_FAILURE();
870             SetLastError(ERROR_INVALID_HANDLE);
871             hMem = NULL;
872         }
873         else if (uFlags & GMEM_MODIFY)
874         {
875             /* User is changing flags... check if the memory was discardable */
876             if (uFlags & GMEM_DISCARDABLE)
877             {
878                 /* Then set the flag */
879                 HandleEntry->Flags |= BASE_HEAP_ENTRY_FLAG_REUSABLE;
880             }
881             else
882             {
883                 /* Otherwise, remove the flag */
884                 HandleEntry->Flags &= ~BASE_HEAP_ENTRY_FLAG_REUSABLE;
885             }
886         }
887         else
888         {
889             /* Otherwise, get the object and check if we have no size */
890             Ptr = HandleEntry->Object;
891             if (!dwBytes)
892             {
893                 /* Clear the handle and check for a pointer */
894                 hMem = NULL;
895                 if (Ptr)
896                 {
897                     /* Make sure the handle isn't locked */
898                     if ((uFlags & GMEM_MOVEABLE) && !(HandleEntry->LockCount))
899                     {
900                         /* Free the current heap */
901                         if (RtlFreeHeap(BaseHeap, Flags, Ptr))
902                         {
903                             /* Free the handle */
904                             HandleEntry->Object = NULL;
905                             HandleEntry->Flags |= BASE_HEAP_ENTRY_FLAG_REUSE;
906 
907                             /* Get the object pointer */
908                             hMem = &HandleEntry->Object;
909                         }
910                     }
911                 }
912                 else
913                 {
914                     /* Otherwise just return the object pointer */
915                     hMem = &HandleEntry->Object;
916                 }
917             }
918             else
919             {
920                 /* Otherwise, we're allocating, so set the new flags needed */
921                 Flags |= HEAP_SETTABLE_USER_VALUE | BASE_HEAP_FLAG_MOVABLE;
922                 if (!Ptr)
923                 {
924                     /* We don't have a base, so allocate one */
925                     Ptr = RtlAllocateHeap(BaseHeap, Flags, dwBytes);
926                     BASE_TRACE_ALLOC2(Ptr);
927                     if (Ptr)
928                     {
929                         /* Allocation succeeded, so save our entry */
930                         RtlSetUserValueHeap(BaseHeap,
931                                             HEAP_NO_SERIALIZE,
932                                             Ptr,
933                                             hMem);
934                     }
935                 }
936                 else
937                 {
938                     /*
939                      * If it's not movable or currently locked, we MUST allocate
940                      * in-place!
941                      */
942                     if (!(uFlags & GMEM_MOVEABLE) && (HandleEntry->LockCount))
943                     {
944                         /* Set the flag */
945                         Flags |= HEAP_REALLOC_IN_PLACE_ONLY;
946                     }
947                     else
948                     {
949                         /* Otherwise clear the flag if we set it previously */
950                         Flags &= ~HEAP_REALLOC_IN_PLACE_ONLY;
951                     }
952 
953                     /* Do the re-allocation. No need to save the entry again */
954                     Ptr = RtlReAllocateHeap(BaseHeap, Flags, Ptr, dwBytes);
955                 }
956 
957                 /* Make sure we have a pointer by now */
958                 if (Ptr)
959                 {
960                     /* Write it in the handle entry and mark it in use */
961                     HandleEntry->Object = Ptr;
962                     HandleEntry->Flags &= ~BASE_HEAP_ENTRY_FLAG_REUSE;
963                 }
964                 else
965                 {
966                     /* Otherwise we failed */
967                     hMem = NULL;
968                     SetLastError(ERROR_NOT_ENOUGH_MEMORY);
969                 }
970             }
971         }
972     }
973     else if (uFlags & GMEM_MODIFY)
974     {
975         /* This is not a handle-based heap and the caller wants it to be one */
976         if (uFlags & GMEM_MOVEABLE)
977         {
978             /* Get information on its current state */
979             Handle = hMem;
980             if (RtlGetUserInfoHeap(BaseHeap,
981                                    HEAP_NO_SERIALIZE,
982                                    hMem,
983                                    &Handle,
984                                    NULL))
985             {
986                 /*
987                  * Check if the handle matches the pointer or the moveable flag
988                  * isn't there, which is what we expect since it currently isn't.
989                  */
990                 if ((Handle == hMem) || !(Flags & BASE_HEAP_FLAG_MOVABLE))
991                 {
992                     /* Allocate a handle for it */
993                     HandleEntry = BaseHeapAllocEntry();
994                     if (!HandleEntry)
995                     {
996                         /* No entry could be allocated */
997                         SetLastError(ERROR_NOT_ENOUGH_MEMORY);
998                         RtlUnlockHeap(BaseHeap);
999                         return NULL;
1000                     }
1001 
1002                     /* Calculate the size of the current heap */
1003                     dwBytes = RtlSizeHeap(BaseHeap, HEAP_NO_SERIALIZE, hMem);
1004 
1005                     /* Set the movable flag */
1006                     Flags |= HEAP_SETTABLE_USER_VALUE | BASE_HEAP_FLAG_MOVABLE;
1007 
1008                     /* Now allocate the actual heap for it */
1009                     HandleEntry->Object = RtlAllocateHeap(BaseHeap,
1010                                                           Flags,
1011                                                           dwBytes);
1012                     BASE_TRACE_PTR(HandleEntry->Object, HandleEntry);
1013                     if (!HandleEntry->Object)
1014                     {
1015                         /*
1016                          * We failed, manually set the allocate flag and
1017                          * free the handle
1018                          */
1019                         HandleEntry->Flags = RTL_HANDLE_VALID;
1020                         BaseHeapFreeEntry(HandleEntry);
1021 
1022                         /* For the cleanup case */
1023                         BASE_TRACE_FAILURE();
1024                         HandleEntry = NULL;
1025                         SetLastError(ERROR_NOT_ENOUGH_MEMORY);
1026                     }
1027                     else
1028                     {
1029                         /* Otherwise, copy the new heap and free the old one */
1030                         RtlMoveMemory(HandleEntry->Object, hMem, dwBytes);
1031                         RtlFreeHeap(BaseHeap, HEAP_NO_SERIALIZE, hMem);
1032 
1033                         /* Select the heap pointer */
1034                         hMem = (HANDLE)&HandleEntry->Object;
1035 
1036                         /* Initialize the count and default flags */
1037                         HandleEntry->LockCount = 0;
1038                         HandleEntry->Flags = RTL_HANDLE_VALID |
1039                                              BASE_HEAP_ENTRY_FLAG_MOVABLE;
1040 
1041                         /* Check if it's also discardable */
1042                         if (uFlags & GMEM_DISCARDABLE)
1043                         {
1044                             /* Set the internal flag */
1045                             HandleEntry->Flags |= BASE_HEAP_ENTRY_FLAG_REUSABLE;
1046                         }
1047 
1048                         /* Check if it's also DDE Shared */
1049                         if (uFlags & GMEM_DDESHARE)
1050                         {
1051                             /* Set the internal flag */
1052                             HandleEntry->Flags |= BASE_HEAP_ENTRY_FLAG_DDESHARE;
1053                         }
1054 
1055                         /* Allocation succeeded, so save our entry */
1056                         RtlSetUserValueHeap(BaseHeap,
1057                                             HEAP_NO_SERIALIZE,
1058                                             HandleEntry->Object,
1059                                             hMem);
1060                     }
1061                 }
1062             }
1063         }
1064     }
1065     else
1066     {
1067         /* Otherwise, this is a simple RTL Managed Heap, so just call it */
1068         hMem = RtlReAllocateHeap(BaseHeap,
1069                                  Flags | HEAP_NO_SERIALIZE,
1070                                  hMem,
1071                                  dwBytes);
1072         if (!hMem)
1073         {
1074             /* Fail */
1075             BASE_TRACE_FAILURE();
1076             SetLastError(ERROR_NOT_ENOUGH_MEMORY);
1077         }
1078     }
1079 
1080     /* All done, unlock the heap and return the pointer */
1081     RtlUnlockHeap(BaseHeap);
1082     return hMem;
1083 }
1084 
1085 /*
1086  * @implemented
1087  */
1088 SIZE_T
1089 NTAPI
1090 GlobalSize(HGLOBAL hMem)
1091 {
1092     PBASE_HEAP_HANDLE_ENTRY HandleEntry;
1093     PVOID Handle = NULL;
1094     ULONG Flags = 0;
1095     SIZE_T dwSize = MAXULONG_PTR;
1096 
1097     /* Lock the heap */
1098     RtlLockHeap(BaseHeap);
1099     _SEH2_TRY
1100     {
1101         /* Check if this is a simple RTL Heap Managed block */
1102         if (!((ULONG_PTR)hMem & BASE_HEAP_IS_HANDLE_ENTRY))
1103         {
1104             /* Then we'll query RTL Heap */
1105             if (RtlGetUserInfoHeap(BaseHeap, Flags, hMem, &Handle, &Flags))
1106             {
1107                 BASE_TRACE_PTR(Handle, hMem);
1108                 /*
1109                  * Check if RTL Heap didn't give us a handle or said that this
1110                  * heap isn't movable.
1111                  */
1112                 if (!(Handle) || !(Flags & BASE_HEAP_FLAG_MOVABLE))
1113                 {
1114                     /* We're not a handle heap, so use the generic call */
1115                     dwSize = RtlSizeHeap(BaseHeap, HEAP_NO_SERIALIZE, hMem);
1116                 }
1117                 else
1118                 {
1119                     /* We're a handle heap so get the internal handle */
1120                     hMem = Handle;
1121                 }
1122             }
1123         }
1124 
1125         /* Make sure that this is an entry in our handle database */
1126         if ((ULONG_PTR)hMem & BASE_HEAP_IS_HANDLE_ENTRY)
1127         {
1128             /* Get the entry */
1129             HandleEntry = BaseHeapGetEntry(hMem);
1130             BASE_TRACE_HANDLE(HandleEntry, hMem);
1131 
1132             /* Make sure the handle is valid */
1133             if (!BaseHeapValidateEntry(HandleEntry))
1134             {
1135                 /* Fail */
1136                 BASE_TRACE_FAILURE();
1137                 SetLastError(ERROR_INVALID_HANDLE);
1138             }
1139             else if (HandleEntry->Flags & BASE_HEAP_ENTRY_FLAG_REUSE)
1140             {
1141                 /* We've reused this block, but we've saved the size for you */
1142                 dwSize = HandleEntry->OldSize;
1143             }
1144             else
1145             {
1146                 /* Otherwise, query RTL about it */
1147                 dwSize = RtlSizeHeap(BaseHeap,
1148                                      HEAP_NO_SERIALIZE,
1149                                      HandleEntry->Object);
1150             }
1151         }
1152     }
1153     _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
1154     {
1155         /* Set failure for later */
1156         dwSize = MAXULONG_PTR;
1157     }
1158     _SEH2_END;
1159 
1160     /* Check if by now, we still haven't gotten any useful size */
1161     if (dwSize == MAXULONG_PTR)
1162     {
1163         /* Fail */
1164         BASE_TRACE_FAILURE();
1165         SetLastError(ERROR_INVALID_HANDLE);
1166         dwSize = 0;
1167     }
1168 
1169     /* All done! Unlock heap and return the size */
1170     RtlUnlockHeap(BaseHeap);
1171     return dwSize;
1172 }
1173 
1174 /*
1175  * @implemented
1176  */
1177 VOID
1178 NTAPI
1179 GlobalUnfix(HGLOBAL hMem)
1180 {
1181     /* If the handle is valid, unlock it */
1182     if (hMem != INVALID_HANDLE_VALUE) GlobalUnlock(hMem);
1183 }
1184 
1185 /*
1186  * @implemented
1187  */
1188 BOOL
1189 NTAPI
1190 GlobalUnlock(HGLOBAL hMem)
1191 {
1192     PBASE_HEAP_HANDLE_ENTRY HandleEntry;
1193     BOOL RetVal = TRUE;
1194 
1195     /* Check if this was a simple allocated heap entry */
1196     if (!((ULONG_PTR)hMem & BASE_HEAP_IS_HANDLE_ENTRY)) return RetVal;
1197 
1198     /* Otherwise, lock the heap */
1199     RtlLockHeap(BaseHeap);
1200 
1201     /* Get the handle entry */
1202     HandleEntry = BaseHeapGetEntry(hMem);
1203     BASE_TRACE_HANDLE(HandleEntry, hMem);
1204 
1205     _SEH2_TRY
1206     {
1207         /* Make sure it's valid */
1208         if (!BaseHeapValidateEntry(HandleEntry))
1209         {
1210             /* It's not, fail */
1211             BASE_TRACE_FAILURE();
1212             SetLastError(ERROR_INVALID_HANDLE);
1213             RetVal = FALSE;
1214         }
1215         else
1216         {
1217             /* Otherwise, decrement lock count, unless we're already at 0*/
1218             if (!HandleEntry->LockCount--)
1219             {
1220                 /* In which case we simply lock it back and fail */
1221                 HandleEntry->LockCount++;
1222                 SetLastError(ERROR_NOT_LOCKED);
1223                 RetVal = FALSE;
1224             }
1225             else if (!HandleEntry->LockCount)
1226             {
1227                 /* Nothing to unlock */
1228                 SetLastError(NO_ERROR);
1229                 RetVal = FALSE;
1230             }
1231         }
1232     }
1233     _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
1234     {
1235         SetLastError(ERROR_INVALID_PARAMETER);
1236         RetVal = FALSE;
1237     }
1238     _SEH2_END;
1239 
1240     /* All done. Unlock the heap and return the pointer */
1241     RtlUnlockHeap(BaseHeap);
1242     return RetVal;
1243 }
1244 
1245 /*
1246  * @implemented
1247  */
1248 BOOL
1249 NTAPI
1250 GlobalUnWire(HGLOBAL hMem)
1251 {
1252     /* This is simply an unlock */
1253     return GlobalUnlock(hMem);
1254 }
1255 
1256 /*
1257  * @implemented
1258  */
1259 LPVOID
1260 NTAPI
1261 GlobalWire(HGLOBAL hMem)
1262 {
1263     /* This is just a lock */
1264     return GlobalLock(hMem);
1265 }
1266 
1267 /*
1268  * @implemented
1269  */
1270 BOOL
1271 NTAPI
1272 GlobalMemoryStatusEx(LPMEMORYSTATUSEX lpBuffer)
1273 {
1274     SYSTEM_PERFORMANCE_INFORMATION PerformanceInfo;
1275     VM_COUNTERS VmCounters;
1276     QUOTA_LIMITS QuotaLimits;
1277     ULONGLONG PageFile, PhysicalMemory;
1278 
1279     if (lpBuffer->dwLength != sizeof(*lpBuffer))
1280     {
1281         SetLastError(ERROR_INVALID_PARAMETER);
1282         return FALSE;
1283     }
1284 
1285     /* Query performance information */
1286     NtQuerySystemInformation(SystemPerformanceInformation,
1287                              &PerformanceInfo,
1288                              sizeof(PerformanceInfo),
1289                              NULL);
1290 
1291     /* Calculate memory load */
1292     lpBuffer->dwMemoryLoad = ((DWORD)(BaseStaticServerData->SysInfo.NumberOfPhysicalPages -
1293                                       PerformanceInfo.AvailablePages) * 100) /
1294                                       BaseStaticServerData->SysInfo.NumberOfPhysicalPages;
1295 
1296     /* Save physical memory */
1297     PhysicalMemory = BaseStaticServerData->SysInfo.NumberOfPhysicalPages *
1298                      BaseStaticServerData->SysInfo.PageSize;
1299     lpBuffer->ullTotalPhys = PhysicalMemory;
1300 
1301     /* Now save available physical memory */
1302     PhysicalMemory = PerformanceInfo.AvailablePages *
1303                      BaseStaticServerData->SysInfo.PageSize;
1304     lpBuffer->ullAvailPhys = PhysicalMemory;
1305 
1306     /* Query VM and Quota Limits */
1307     NtQueryInformationProcess(NtCurrentProcess(),
1308                               ProcessQuotaLimits,
1309                               &QuotaLimits,
1310                               sizeof(QUOTA_LIMITS),
1311                               NULL);
1312     NtQueryInformationProcess(NtCurrentProcess(),
1313                               ProcessVmCounters,
1314                               &VmCounters,
1315                               sizeof(VM_COUNTERS),
1316                               NULL);
1317 
1318     /* Save the commit limit */
1319     lpBuffer->ullTotalPageFile = min(QuotaLimits.PagefileLimit,
1320                                      PerformanceInfo.CommitLimit);
1321     lpBuffer->ullTotalPageFile *= BaseStaticServerData->SysInfo.PageSize;
1322 
1323     /* Calculate how many pages are left */
1324     PageFile = PerformanceInfo.CommitLimit - PerformanceInfo.CommittedPages;
1325 
1326     /* Save the total */
1327     lpBuffer->ullAvailPageFile = min(PageFile,
1328                                      QuotaLimits.PagefileLimit -
1329                                      VmCounters.PagefileUsage);
1330     lpBuffer->ullAvailPageFile *= BaseStaticServerData->SysInfo.PageSize;
1331 
1332     /* Now calculate the total virtual space */
1333     lpBuffer->ullTotalVirtual = (BaseStaticServerData->SysInfo.MaximumUserModeAddress -
1334                                  BaseStaticServerData->SysInfo.MinimumUserModeAddress) + 1;
1335 
1336     /* And finally the available virtual space */
1337     lpBuffer->ullAvailVirtual = lpBuffer->ullTotalVirtual - VmCounters.VirtualSize;
1338     lpBuffer->ullAvailExtendedVirtual = 0;
1339 
1340     return TRUE;
1341 }
1342 
1343 /*
1344  * @implemented
1345  */
1346 VOID
1347 NTAPI
1348 GlobalMemoryStatus(LPMEMORYSTATUS lpBuffer)
1349 {
1350     MEMORYSTATUSEX lpBufferEx;
1351 
1352     /* Call the extended function */
1353     lpBufferEx.dwLength = sizeof(MEMORYSTATUSEX);
1354     if (GlobalMemoryStatusEx(&lpBufferEx))
1355     {
1356         /* Reset the right size and fill out the information */
1357         lpBuffer->dwLength = sizeof(MEMORYSTATUS);
1358         lpBuffer->dwMemoryLoad = lpBufferEx.dwMemoryLoad;
1359         lpBuffer->dwTotalPhys = (SIZE_T)min(lpBufferEx.ullTotalPhys, MAXULONG_PTR);
1360         lpBuffer->dwAvailPhys = (SIZE_T)min(lpBufferEx.ullAvailPhys, MAXULONG_PTR);
1361         lpBuffer->dwTotalPageFile = (SIZE_T)min(lpBufferEx.ullTotalPageFile, MAXULONG_PTR);
1362         lpBuffer->dwAvailPageFile = (SIZE_T)min(lpBufferEx.ullAvailPageFile, MAXULONG_PTR);
1363         lpBuffer->dwTotalVirtual = (SIZE_T)min(lpBufferEx.ullTotalVirtual, MAXULONG_PTR);
1364         lpBuffer->dwAvailVirtual = (SIZE_T)min(lpBufferEx.ullAvailVirtual, MAXULONG_PTR);
1365     }
1366 }
1367 
1368 /*
1369  * @implemented
1370  */
1371 HLOCAL
1372 NTAPI
1373 LocalAlloc(UINT uFlags,
1374            SIZE_T dwBytes)
1375 {
1376     ULONG Flags = 0;
1377     PVOID Ptr = NULL;
1378     HANDLE hMemory;
1379     PBASE_HEAP_HANDLE_ENTRY HandleEntry;
1380     BASE_TRACE_ALLOC(dwBytes, uFlags);
1381     ASSERT(BaseHeap);
1382 
1383     /* Make sure the flags are valid */
1384     if (uFlags & ~LMEM_VALID_FLAGS)
1385     {
1386         /* They aren't, fail */
1387         BASE_TRACE_FAILURE();
1388         SetLastError(ERROR_INVALID_PARAMETER);
1389         return NULL;
1390     }
1391 
1392     /* Convert ZEROINIT */
1393     if (uFlags & LMEM_ZEROINIT) Flags |= HEAP_ZERO_MEMORY;
1394 
1395     /* Check if we're not movable, which means pointer-based heap */
1396     if (!(uFlags & LMEM_MOVEABLE))
1397     {
1398         /* Allocate heap for it */
1399         Ptr = RtlAllocateHeap(BaseHeap, Flags, dwBytes);
1400         BASE_TRACE_ALLOC2(Ptr);
1401         return Ptr;
1402     }
1403 
1404     /* This is heap based, so lock it in first */
1405     RtlLockHeap(BaseHeap);
1406 
1407     /*
1408      * Disable locking, enable custom flags, and write the
1409      * movable flag (deprecated)
1410      */
1411     Flags |= HEAP_NO_SERIALIZE |
1412              HEAP_SETTABLE_USER_VALUE |
1413              BASE_HEAP_FLAG_MOVABLE;
1414 
1415     /* Allocate the handle */
1416     HandleEntry = BaseHeapAllocEntry();
1417     if (!HandleEntry)
1418     {
1419         /* Fail */
1420         hMemory = NULL;
1421         SetLastError(ERROR_NOT_ENOUGH_MEMORY);
1422         BASE_TRACE_FAILURE();
1423         goto Quickie;
1424     }
1425 
1426     /* Get the object and make sure we have size */
1427     hMemory = &HandleEntry->Object;
1428     if (dwBytes)
1429     {
1430         /* Allocate the actual memory for it */
1431         Ptr = RtlAllocateHeap(BaseHeap, Flags, dwBytes);
1432         BASE_TRACE_PTR(HandleEntry, Ptr);
1433         if (!Ptr)
1434         {
1435             /* We failed, manually set the allocate flag and free the handle */
1436             HandleEntry->Flags = RTL_HANDLE_VALID;
1437             BaseHeapFreeEntry(HandleEntry);
1438 
1439             /* For the cleanup case */
1440             HandleEntry = NULL;
1441         }
1442         else
1443         {
1444             /* All worked well, save our heap entry */
1445             RtlSetUserValueHeap(BaseHeap, HEAP_NO_SERIALIZE, Ptr, hMemory);
1446         }
1447     }
1448 
1449 Quickie:
1450     /* Cleanup! First unlock the heap */
1451     RtlUnlockHeap(BaseHeap);
1452 
1453     /* Check if a handle was allocated */
1454     if (HandleEntry)
1455     {
1456         /* Set the pointer and allocated flag */
1457         HandleEntry->Object = Ptr;
1458         HandleEntry->Flags = RTL_HANDLE_VALID;
1459         if (!Ptr)
1460         {
1461             /* We don't have a valid pointer, but so reuse this handle */
1462             HandleEntry->Flags |= BASE_HEAP_ENTRY_FLAG_REUSE;
1463         }
1464 
1465         /* Check if the handle is discardable */
1466         if (uFlags & GMEM_DISCARDABLE)
1467         {
1468             /* Save it in the handle entry */
1469             HandleEntry->Flags |= BASE_HEAP_ENTRY_FLAG_REUSABLE;
1470         }
1471 
1472         /* Check if the handle is moveable */
1473         if (uFlags & GMEM_MOVEABLE)
1474         {
1475             /* Save it in the handle entry */
1476             HandleEntry->Flags |= BASE_HEAP_ENTRY_FLAG_MOVABLE;
1477         }
1478 
1479         /* Set the pointer */
1480         Ptr = hMemory;
1481     }
1482 
1483     /* Return the pointer */
1484     return Ptr;
1485 }
1486 
1487 /*
1488  * @implemented
1489  */
1490 SIZE_T
1491 NTAPI
1492 LocalCompact(UINT dwMinFree)
1493 {
1494     /* Call the RTL Heap Manager */
1495     return RtlCompactHeap(BaseHeap, 0);
1496 }
1497 
1498 /*
1499  * @implemented
1500  */
1501 UINT
1502 NTAPI
1503 LocalFlags(HLOCAL hMem)
1504 {
1505     PBASE_HEAP_HANDLE_ENTRY HandleEntry;
1506     HANDLE Handle = NULL;
1507     ULONG Flags = 0;
1508     UINT uFlags = LMEM_INVALID_HANDLE;
1509 
1510     /* Start by locking the heap */
1511     RtlLockHeap(BaseHeap);
1512 
1513     /* Check if this is a simple RTL Heap Managed block */
1514     if (!((ULONG_PTR)hMem & BASE_HEAP_IS_HANDLE_ENTRY))
1515     {
1516         /* Then we'll query RTL Heap */
1517         RtlGetUserInfoHeap(BaseHeap, Flags, hMem, &Handle, &Flags);
1518         BASE_TRACE_PTR(Handle, hMem);
1519 
1520         /*
1521          * Check if RTL Heap didn't find a handle associated with us or
1522          * said that this heap isn't movable, which means something we're
1523          * really not a handle-based heap.
1524          */
1525         if (!(Handle) || !(Flags & BASE_HEAP_FLAG_MOVABLE))
1526         {
1527             /* Then set the flags to 0 */
1528             uFlags = 0;
1529         }
1530         else
1531         {
1532             /* Otherwise we're handle-based, so get the internal handle */
1533             hMem = Handle;
1534         }
1535     }
1536 
1537     /* Check if the handle is actually an entry in our table */
1538     if ((ULONG_PTR)hMem & BASE_HEAP_IS_HANDLE_ENTRY)
1539     {
1540         /* Then get the entry */
1541         HandleEntry = BaseHeapGetEntry(hMem);
1542         BASE_TRACE_HANDLE(HandleEntry, hMem);
1543 
1544         /* Make sure it's a valid handle */
1545         if (BaseHeapValidateEntry(HandleEntry))
1546         {
1547             /* Get the lock count first */
1548             uFlags = HandleEntry->LockCount & LMEM_LOCKCOUNT;
1549 
1550             /* Now check if it's discardable */
1551             if (HandleEntry->Flags & BASE_HEAP_ENTRY_FLAG_REUSABLE)
1552             {
1553                 /* Set the Win32 Flag */
1554                 uFlags |= LMEM_DISCARDABLE;
1555             }
1556 
1557             /* Now check if it's discarded */
1558             if (HandleEntry->Flags & BASE_HEAP_ENTRY_FLAG_REUSE)
1559                /* Set the Win32 Flag */
1560                uFlags |= LMEM_DISCARDED;
1561         }
1562     }
1563 
1564     /* Check if by now, we still haven't gotten any useful flags */
1565     if (uFlags == LMEM_INVALID_HANDLE) SetLastError(ERROR_INVALID_HANDLE);
1566 
1567     /* All done! Unlock heap and return Win32 Flags */
1568     RtlUnlockHeap(BaseHeap);
1569     return uFlags;
1570 }
1571 
1572 /*
1573  * @implemented
1574  */
1575 HLOCAL
1576 NTAPI
1577 LocalFree(HLOCAL hMem)
1578 {
1579     /* This is identical to a Global Free */
1580     return GlobalFree(hMem);
1581 }
1582 
1583 /*
1584  * @implemented
1585  */
1586 HLOCAL
1587 NTAPI
1588 LocalHandle(LPCVOID pMem)
1589 {
1590     /* This is identical to a Global Handle */
1591     return GlobalHandle(pMem);
1592 }
1593 
1594 /*
1595  * @implemented
1596  */
1597 LPVOID
1598 NTAPI
1599 LocalLock(HLOCAL hMem)
1600 {
1601     /* This is the same as a GlobalLock, assuming these never change */
1602     C_ASSERT(LMEM_LOCKCOUNT == GMEM_LOCKCOUNT);
1603     return GlobalLock(hMem);
1604 }
1605 
1606 HLOCAL
1607 NTAPI
1608 LocalReAlloc(HLOCAL hMem,
1609              SIZE_T dwBytes,
1610              UINT uFlags)
1611 {
1612     PBASE_HEAP_HANDLE_ENTRY HandleEntry;
1613     LPVOID Ptr;
1614     ULONG Flags = 0;
1615 
1616     /* Convert ZEROINIT */
1617     if (uFlags & LMEM_ZEROINIT) Flags |= HEAP_ZERO_MEMORY;
1618 
1619     /* If this wasn't a movable heap, then we MUST re-alloc in place */
1620     if (!(uFlags & LMEM_MOVEABLE)) Flags |= HEAP_REALLOC_IN_PLACE_ONLY;
1621 
1622     /* Lock the heap and disable built-in locking in the RTL Heap functions */
1623     RtlLockHeap(BaseHeap);
1624     Flags |= HEAP_NO_SERIALIZE;
1625 
1626     /* Check if this is a simple handle-based block */
1627     if (((ULONG_PTR)hMem & BASE_HEAP_IS_HANDLE_ENTRY))
1628     {
1629         /* Get the entry */
1630         HandleEntry = BaseHeapGetEntry(hMem);
1631         BASE_TRACE_HANDLE(HandleEntry, hMem);
1632 
1633         /* Make sure the handle is valid */
1634         if (!BaseHeapValidateEntry(HandleEntry))
1635         {
1636             /* Fail */
1637             BASE_TRACE_FAILURE();
1638             SetLastError(ERROR_INVALID_HANDLE);
1639             hMem = NULL;
1640         }
1641         else if (uFlags & LMEM_MODIFY)
1642         {
1643             /* User is changing flags... check if the memory was discardable */
1644             if (uFlags & LMEM_DISCARDABLE)
1645             {
1646                 /* Then set the flag */
1647                 HandleEntry->Flags |= BASE_HEAP_ENTRY_FLAG_REUSABLE;
1648             }
1649             else
1650             {
1651                 /* Otherwise, remove the flag */
1652                 HandleEntry->Flags &= ~BASE_HEAP_ENTRY_FLAG_REUSABLE;
1653             }
1654         }
1655         else
1656         {
1657             /* Otherwise, get the object and check if we have no size */
1658             Ptr = HandleEntry->Object;
1659             if (!dwBytes)
1660             {
1661                 /* Clear the handle and check for a pointer */
1662                 hMem = NULL;
1663                 if (Ptr)
1664                 {
1665                     /* Make sure the handle isn't locked */
1666                     if ((uFlags & LMEM_MOVEABLE) && !(HandleEntry->LockCount))
1667                     {
1668                         /* Free the current heap */
1669                         RtlFreeHeap(BaseHeap, Flags, Ptr);
1670 
1671                         /* Free the handle */
1672                         HandleEntry->Object = NULL;
1673                         HandleEntry->Flags |= BASE_HEAP_ENTRY_FLAG_REUSE;
1674 
1675                         /* Get the object pointer */
1676                         hMem = &HandleEntry->Object;
1677                     }
1678                 }
1679                 else
1680                 {
1681                     /* Otherwise just return the object pointer */
1682                     hMem = &HandleEntry->Object;
1683                 }
1684             }
1685             else
1686             {
1687                 /* Otherwise, we're allocating, so set the new flags needed */
1688                 Flags |= HEAP_SETTABLE_USER_VALUE | BASE_HEAP_FLAG_MOVABLE;
1689                 if (!Ptr)
1690                 {
1691                     /* We don't have a base, so allocate one */
1692                     Ptr = RtlAllocateHeap(BaseHeap, Flags, dwBytes);
1693                     BASE_TRACE_ALLOC2(Ptr);
1694                     if (Ptr)
1695                     {
1696                         /* Allocation succeeded, so save our entry */
1697                         RtlSetUserValueHeap(BaseHeap,
1698                                             HEAP_NO_SERIALIZE,
1699                                             Ptr,
1700                                             hMem);
1701                     }
1702                 }
1703                 else
1704                 {
1705                     /*
1706                      * If it's not movable or currently locked, we MUST allocate
1707                      * in-place!
1708                      */
1709                     if (!(uFlags & LMEM_MOVEABLE) && (HandleEntry->LockCount))
1710                     {
1711                         /* Set the flag */
1712                         Flags |= HEAP_REALLOC_IN_PLACE_ONLY;
1713                     }
1714                     else
1715                     {
1716                         /* Otherwise clear the flag if we set it previously */
1717                         Flags &= ~HEAP_REALLOC_IN_PLACE_ONLY;
1718                     }
1719 
1720                     /* And do the re-allocation */
1721                     Ptr = RtlReAllocateHeap(BaseHeap, Flags, Ptr, dwBytes);
1722                 }
1723 
1724                 /* Make sure we have a pointer by now */
1725                 if (Ptr)
1726                 {
1727                     /* Write it in the handle entry and mark it in use */
1728                     HandleEntry->Object = Ptr;
1729                     HandleEntry->Flags &= ~BASE_HEAP_ENTRY_FLAG_REUSE;
1730                 }
1731                 else
1732                 {
1733                     /* Otherwise we failed */
1734                     hMem = NULL;
1735                     SetLastError(ERROR_NOT_ENOUGH_MEMORY);
1736                 }
1737             }
1738         }
1739     }
1740     else if (!(uFlags & LMEM_MODIFY))
1741     {
1742         /* Otherwise, this is a simple RTL Managed Heap, so just call it */
1743         hMem = RtlReAllocateHeap(BaseHeap,
1744                                  Flags | HEAP_NO_SERIALIZE,
1745                                  hMem,
1746                                  dwBytes);
1747         if (!hMem)
1748         {
1749             /* Fail */
1750             BASE_TRACE_FAILURE();
1751             SetLastError(ERROR_NOT_ENOUGH_MEMORY);
1752         }
1753     }
1754 
1755     /* All done, unlock the heap and return the pointer */
1756     RtlUnlockHeap(BaseHeap);
1757     return hMem;
1758 }
1759 
1760 /*
1761  * @implemented
1762  */
1763 SIZE_T
1764 WINAPI
1765 LocalShrink(HLOCAL hMem,
1766             UINT cbNewSize)
1767 {
1768     /* Call RTL */
1769     return RtlCompactHeap(BaseHeap, 0);
1770 }
1771 
1772 /*
1773  * @implemented
1774  */
1775 SIZE_T
1776 NTAPI
1777 LocalSize(HLOCAL hMem)
1778 {
1779     /* This is the same as a Global Size */
1780     return GlobalSize(hMem);
1781 }
1782 
1783 /*
1784  * @implemented
1785  */
1786 BOOL
1787 NTAPI
1788 LocalUnlock(HLOCAL hMem)
1789 {
1790     PBASE_HEAP_HANDLE_ENTRY HandleEntry;
1791     BOOL RetVal = TRUE;
1792 
1793     /* Check if this was a simple allocated heap entry */
1794     if (!((ULONG_PTR)hMem & BASE_HEAP_IS_HANDLE_ENTRY))
1795     {
1796        /* Fail, because LocalUnlock is not supported on LMEM_FIXED allocations */
1797        SetLastError(ERROR_NOT_LOCKED);
1798        return FALSE;
1799     }
1800 
1801     /* Otherwise, lock the heap */
1802     RtlLockHeap(BaseHeap);
1803 
1804     /* Get the handle entry */
1805     HandleEntry = BaseHeapGetEntry(hMem);
1806     BASE_TRACE_HANDLE(HandleEntry, hMem);
1807     _SEH2_TRY
1808     {
1809         /* Make sure it's valid */
1810         if (!BaseHeapValidateEntry(HandleEntry))
1811         {
1812             /* It's not, fail */
1813             BASE_TRACE_FAILURE();
1814             SetLastError(ERROR_INVALID_HANDLE);
1815             RetVal = FALSE;
1816         }
1817         else
1818         {
1819             /* Otherwise, decrement lock count, unless we're already at 0*/
1820             if (!HandleEntry->LockCount--)
1821             {
1822                 /* In which case we simply lock it back and fail */
1823                 HandleEntry->LockCount++;
1824                 SetLastError(ERROR_NOT_LOCKED);
1825                 RetVal = FALSE;
1826             }
1827             else if (!HandleEntry->LockCount)
1828             {
1829                 /* Nothing to unlock */
1830                 SetLastError(NO_ERROR);
1831                 RetVal = FALSE;
1832             }
1833         }
1834     }
1835     _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
1836     {
1837         SetLastError(ERROR_INVALID_PARAMETER);
1838         RetVal = FALSE;
1839     }
1840     _SEH2_END;
1841 
1842     /* All done. Unlock the heap and return the pointer */
1843     RtlUnlockHeap(BaseHeap);
1844     return RetVal;
1845 }
1846 
1847 /* EOF */
1848