ReactOS 0.4.16-dev-2354-g16de117
MmMdl.c
Go to the documentation of this file.
1/*
2 * PROJECT: ReactOS kernel-mode tests
3 * LICENSE: LGPL-2.1-or-later (https://spdx.org/licenses/LGPL-2.1-or-later)
4 * PURPOSE: Kernel-Mode Test Suite MDL test
5 * COPYRIGHT: Copyright 2015,2023 Thomas Faber (thomas.faber@reactos.org)
6 * COPYRIGHT: Copyright 2017 Pierre Schweitzer (pierre@reactos.org)
7 */
8
9#include <kmt_test.h>
10
11static
12VOID
14{
15 PMDL Mdl;
16 PHYSICAL_ADDRESS LowAddress;
19 PVOID SystemVa;
20 PMDL Mdls[32];
21 PVOID SystemVas[32];
22 ULONG i;
23 PPFN_NUMBER MdlPages;
24 ULONG MdlPageCount;
25
26 LowAddress.QuadPart = 0;
29 /* simple allocate/free */
30 Mdl = MmAllocatePagesForMdl(LowAddress,
33 2 * 1024 * 1024);
34 ok(Mdl != NULL, "MmAllocatePagesForMdl failed\n");
35 if (skip(Mdl != NULL, "No Mdl\n"))
36 return;
37 ok(MmGetMdlByteCount(Mdl) == 2 * 1024 * 1024, "Byte count: %lu\n", MmGetMdlByteCount(Mdl));
38 ok(MmGetMdlVirtualAddress(Mdl) == NULL, "Virtual address: %p\n", MmGetMdlVirtualAddress(Mdl));
39 ok(!(Mdl->MdlFlags & MDL_MAPPED_TO_SYSTEM_VA), "MdlFlags: %lx\n", Mdl->MdlFlags);
40 MdlPages = MmGetMdlPfnArray(Mdl);
42 ok(MdlPageCount == 2 * 1024 * 1024 / PAGE_SIZE, "MdlPageCount = %lu\n", MdlPageCount);
43 for (i = 0; i < MdlPageCount; i++)
44 {
45 ok(MdlPages[i] != 0 && MdlPages[i] != (PFN_NUMBER)-1,
46 "MdlPages[%lu] = 0x%I64x\n", i, (ULONGLONG)MdlPages[i]);
47 }
50
51 /* Now map/unmap it */
52 Mdl = MmAllocatePagesForMdl(LowAddress,
55 2 * 1024 * 1024);
56 ok(Mdl != NULL, "MmAllocatePagesForMdl failed\n");
57 if (skip(Mdl != NULL, "No Mdl\n"))
58 return;
59 ok(MmGetMdlByteCount(Mdl) == 2 * 1024 * 1024, "Byte count: %lu\n", MmGetMdlByteCount(Mdl));
60 ok(MmGetMdlVirtualAddress(Mdl) == NULL, "Virtual address: %p\n", MmGetMdlVirtualAddress(Mdl));
61 ok(!(Mdl->MdlFlags & MDL_MAPPED_TO_SYSTEM_VA), "MdlFlags: %lx\n", Mdl->MdlFlags);
65 NULL,
66 FALSE,
68 ok(SystemVa != NULL, "MmMapLockedPagesSpecifyCache failed\n");
69 if (!skip(SystemVa != NULL, "No system VA\n"))
70 {
71 ok(MmGetMdlByteCount(Mdl) == 2 * 1024 * 1024, "Byte count: %lu\n", MmGetMdlByteCount(Mdl));
72 ok(MmGetMdlVirtualAddress(Mdl) == NULL, "Virtual address: %p, System VA: %p\n", MmGetMdlVirtualAddress(Mdl), SystemVa);
73 ok(Mdl->MappedSystemVa == SystemVa, "MappedSystemVa: %p, System VA: %p\n", Mdl->MappedSystemVa, SystemVa);
74 ok((Mdl->MdlFlags & MDL_MAPPED_TO_SYSTEM_VA), "MdlFlags: %lx\n", Mdl->MdlFlags);
75 MmUnmapLockedPages(SystemVa, Mdl);
76 }
77 ok(MmGetMdlByteCount(Mdl) == 2 * 1024 * 1024, "Byte count: %lu\n", MmGetMdlByteCount(Mdl));
78 ok(MmGetMdlVirtualAddress(Mdl) == NULL, "Virtual address: %p\n", MmGetMdlVirtualAddress(Mdl));
79 ok(!(Mdl->MdlFlags & MDL_MAPPED_TO_SYSTEM_VA), "MdlFlags: %lx\n", Mdl->MdlFlags);
82
83 /* Now map it, and free without unmapping */
84 Mdl = MmAllocatePagesForMdl(LowAddress,
87 2 * 1024 * 1024);
88 ok(Mdl != NULL, "MmAllocatePagesForMdl failed\n");
89 if (skip(Mdl != NULL, "No Mdl\n"))
90 return;
91 ok(MmGetMdlByteCount(Mdl) == 2 * 1024 * 1024, "Byte count: %lu\n", MmGetMdlByteCount(Mdl));
92 ok(MmGetMdlVirtualAddress(Mdl) == NULL, "Virtual address: %p\n", MmGetMdlVirtualAddress(Mdl));
93 ok(!(Mdl->MdlFlags & MDL_MAPPED_TO_SYSTEM_VA), "MdlFlags: %lx\n", Mdl->MdlFlags);
97 NULL,
98 FALSE,
100 ok(SystemVa != NULL, "MmMapLockedPagesSpecifyCache failed\n");
101 ok(MmGetMdlByteCount(Mdl) == 2 * 1024 * 1024, "Byte count: %lu\n", MmGetMdlByteCount(Mdl));
102 ok(MmGetMdlVirtualAddress(Mdl) == NULL, "Virtual address: %p, System VA: %p\n", MmGetMdlVirtualAddress(Mdl), SystemVa);
103 ok(Mdl->MappedSystemVa == SystemVa, "MappedSystemVa: %p, System VA: %p\n", Mdl->MappedSystemVa, SystemVa);
104 ok((Mdl->MdlFlags & MDL_MAPPED_TO_SYSTEM_VA), "MdlFlags: %lx\n", Mdl->MdlFlags);
107
108 /* try to allocate 2 GB -- should succeed (possibly with fewer pages) but not map */
109 Mdl = MmAllocatePagesForMdl(LowAddress,
111 SkipBytes,
112 2UL * 1024 * 1024 * 1024);
113 ok(Mdl != NULL, "MmAllocatePagesForMdl failed for 2 GB\n");
114 if (Mdl != NULL)
115 {
116 ok(MmGetMdlByteCount(Mdl) <= 2UL * 1024 * 1024 * 1024, "Byte count: %lu\n", MmGetMdlByteCount(Mdl));
117 ok(MmGetMdlVirtualAddress(Mdl) == NULL, "Virtual address: %p\n", MmGetMdlVirtualAddress(Mdl));
118 ok(!(Mdl->MdlFlags & MDL_MAPPED_TO_SYSTEM_VA), "MdlFlags: %lx\n", Mdl->MdlFlags);
119 MdlPages = MmGetMdlPfnArray(Mdl);
121 ok(MdlPageCount <= 2UL * 1024 * 1024 * 1024 / PAGE_SIZE, "MdlPageCount = %lu\n", MdlPageCount);
122 for (i = 0; i < MdlPageCount; i++)
123 {
124 if (MdlPages[i] == 0 ||
125 MdlPages[i] == (PFN_NUMBER)-1)
126 {
127 ok(0, "MdlPages[%lu] = 0x%I64x\n", i, (ULONGLONG)MdlPages[i]);
128 }
129 }
132 MmCached,
133 NULL,
134 FALSE,
136#ifdef _M_IX86
137 /*
138 * MmAllocatePagesForMdl is allowed to return fewer pages than requested.
139 * Only enforce the x86 mapping expectation if we actually got ~2GB.
140 */
141 if (MmGetMdlByteCount(Mdl) >= (1UL << 31))
142 ok(SystemVa == NULL, "MmMapLockedPagesSpecifyCache succeeded for 2 GB\n");
143 else
144 trace("Skipping 2GB mapping expectation (allocated %lu bytes)\n",
146#endif
147 if (SystemVa != NULL)
148 MmUnmapLockedPages(SystemVa, Mdl);
149 ok(MmGetMdlByteCount(Mdl) <= 2UL * 1024 * 1024 * 1024, "Byte count: %lu\n", MmGetMdlByteCount(Mdl));
150 ok(MmGetMdlVirtualAddress(Mdl) == NULL, "Virtual address: %p\n", MmGetMdlVirtualAddress(Mdl));
151 ok(!(Mdl->MdlFlags & MDL_MAPPED_TO_SYSTEM_VA), "MdlFlags: %lx\n", Mdl->MdlFlags);
154 }
155
156 /* now allocate and map 32 MB Mdls until we fail */
157 for (i = 0; i < sizeof(Mdls) / sizeof(Mdls[0]); i++)
158 {
159 Mdls[i] = MmAllocatePagesForMdl(LowAddress,
161 SkipBytes,
162 32 * 1024 * 1024);
163 if (Mdls[i] == NULL)
164 {
165 trace("MmAllocatePagesForMdl failed with i = %lu\n", i);
166 break;
167 }
168 ok(MmGetMdlVirtualAddress(Mdls[i]) == NULL, "Virtual address: %p\n", MmGetMdlVirtualAddress(Mdls[i]));
169 ok(!(Mdls[i]->MdlFlags & MDL_MAPPED_TO_SYSTEM_VA), "MdlFlags: %lx\n", Mdls[i]->MdlFlags);
170 SystemVas[i] = MmMapLockedPagesSpecifyCache(Mdls[i],
172 MmCached,
173 NULL,
174 FALSE,
176 if (SystemVas[i] == NULL)
177 {
178 ok(MmGetMdlByteCount(Mdls[i]) <= 32 * 1024 * 1024, "Byte count: %lu\n", MmGetMdlByteCount(Mdls[i]));
179 ok(MmGetMdlVirtualAddress(Mdls[i]) == NULL, "Virtual address: %p\n", MmGetMdlVirtualAddress(Mdls[i]));
180 ok(!(Mdls[i]->MdlFlags & MDL_MAPPED_TO_SYSTEM_VA), "MdlFlags: %lx\n", Mdls[i]->MdlFlags);
181 trace("MmMapLockedPagesSpecifyCache failed with i = %lu\n", i);
182 break;
183 }
184 ok(MmGetMdlByteCount(Mdls[i]) <= 32 * 1024 * 1024, "Byte count: %lu\n", MmGetMdlByteCount(Mdls[i]));
185 ok(MmGetMdlVirtualAddress(Mdls[i]) == NULL, "Virtual address: %p, System VA: %p\n", MmGetMdlVirtualAddress(Mdls[i]), SystemVas[i]);
186 ok(Mdls[i]->MappedSystemVa == SystemVas[i], "MappedSystemVa: %p\n", Mdls[i]->MappedSystemVa, SystemVas[i]);
187 ok((Mdls[i]->MdlFlags & MDL_MAPPED_TO_SYSTEM_VA), "MdlFlags: %lx\n", Mdls[i]->MdlFlags);
188 }
189 for (i = 0; i < sizeof(Mdls) / sizeof(Mdls[0]); i++)
190 {
191 if (Mdls[i] == NULL)
192 break;
193 if (SystemVas[i] != NULL)
194 MmUnmapLockedPages(SystemVas[i], Mdls[i]);
195 MmFreePagesFromMdl(Mdls[i]);
196 ExFreePoolWithTag(Mdls[i], 0);
197 if (SystemVas[i] == NULL)
198 break;
199 }
200}
201
202static
203VOID
205{
206 PVOID Page;
207 PMDL Mdl;
208
210 ok(Page != NULL, "ExAllocatePoolWithTag failed\n");
211 if (skip(Page != NULL, "No buffer\n"))
212 return;
213
215 ok(Mdl != NULL, "IoAllocateMdl failed\n");
216 if (skip(Mdl != NULL, "No MDL\n"))
217 return;
218
219 ok((Mdl->MdlFlags & MDL_PAGES_LOCKED) == 0, "MDL locked\n");
220 ok((Mdl->MdlFlags & MDL_SOURCE_IS_NONPAGED_POOL) == 0, "MDL from non paged\n");
221
222 // This fails an assertion on Windows 8+ checked and can bugcheck Windows 10+ free.
224 {
226 ok((Mdl->MdlFlags & MDL_PAGES_LOCKED) == 0, "MDL locked\n");
227 ok((Mdl->MdlFlags & MDL_SOURCE_IS_NONPAGED_POOL) != 0, "MDL from paged\n");
228 }
229
230 IoFreeMdl(Mdl);
231 ExFreePoolWithTag(Page, 'Test');
232
234 ok(Page != NULL, "ExAllocatePoolWithTag failed\n");
235 if (skip(Page != NULL, "No buffer\n"))
236 return;
237
239 ok(Mdl != NULL, "IoAllocateMdl failed\n");
240 if (skip(Mdl != NULL, "No MDL\n"))
241 return;
242
243 ok((Mdl->MdlFlags & MDL_PAGES_LOCKED) == 0, "MDL locked\n");
244 ok((Mdl->MdlFlags & MDL_SOURCE_IS_NONPAGED_POOL) == 0, "MDL from non paged\n");
245
247 ok((Mdl->MdlFlags & MDL_PAGES_LOCKED) == 0, "MDL locked\n");
248 ok((Mdl->MdlFlags & MDL_SOURCE_IS_NONPAGED_POOL) != 0, "MDL from paged\n");
249
250 IoFreeMdl(Mdl);
251 ExFreePoolWithTag(Page, 'Test');
252}
253
255{
258}
static VOID TestMmAllocatePagesForMdl(VOID)
Definition: MmMdl.c:13
static VOID TestMmBuildMdlForNonPagedPool(VOID)
Definition: MmMdl.c:204
ULONG_PTR PFN_NUMBER
#define GetNTVersion()
Definition: apitest.h:17
#define trace
Definition: atltest.h:70
#define ok(value,...)
Definition: atltest.h:57
#define skip(...)
Definition: atltest.h:64
#define START_TEST(x)
Definition: atltest.h:75
#define NULL
Definition: types.h:112
#define FALSE
Definition: types.h:117
#define ExAllocatePoolWithTag(hernya, size, tag)
Definition: env_spec_w32.h:350
#define PAGE_SIZE
Definition: env_spec_w32.h:49
#define NonPagedPool
Definition: env_spec_w32.h:307
#define PagedPool
Definition: env_spec_w32.h:308
#define IoFreeMdl
Definition: fxmdl.h:89
#define IoAllocateMdl
Definition: fxmdl.h:88
GLsizei GLenum const GLvoid GLsizei GLenum GLbyte GLbyte GLbyte GLdouble GLdouble GLdouble GLfloat GLfloat GLfloat GLint GLint GLint GLshort GLshort GLshort GLubyte GLubyte GLubyte GLuint GLuint GLuint GLushort GLushort GLushort GLbyte GLbyte GLbyte GLbyte GLdouble GLdouble GLdouble GLdouble GLfloat GLfloat GLfloat GLfloat GLint GLint GLint GLint GLshort GLshort GLshort GLshort GLubyte GLubyte GLubyte GLubyte GLuint GLuint GLuint GLuint GLushort GLushort GLushort GLushort GLboolean const GLdouble const GLfloat const GLint const GLshort const GLbyte const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLdouble const GLfloat const GLfloat const GLint const GLint const GLshort const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort GLenum GLenum GLenum GLfloat GLenum GLint GLenum GLenum GLenum GLfloat GLenum GLenum GLint GLenum GLfloat GLenum GLint GLint GLushort GLenum GLenum GLfloat GLenum GLenum GLint GLfloat const GLubyte GLenum GLenum GLenum const GLfloat GLenum GLenum const GLint GLenum GLint GLint GLsizei GLsizei GLint GLenum GLenum const GLvoid GLenum GLenum const GLfloat GLenum GLenum const GLint GLenum GLenum const GLdouble GLenum GLenum const GLfloat GLenum GLenum const GLint GLsizei GLuint GLfloat GLuint GLbitfield GLfloat GLint GLuint GLboolean GLenum GLfloat GLenum GLbitfield GLenum GLfloat GLfloat GLint GLint const GLfloat GLenum GLfloat GLfloat GLint GLint GLfloat GLfloat GLint GLint const GLfloat GLint GLfloat GLfloat GLint GLfloat GLfloat GLint GLfloat GLfloat const GLdouble const GLfloat const GLdouble const GLfloat GLint i
Definition: glfuncs.h:248
VOID NTAPI MmFreePagesFromMdl(IN PMDL Mdl)
Definition: mdlsup.c:564
PMDL NTAPI MmAllocatePagesForMdl(IN PHYSICAL_ADDRESS LowAddress, IN PHYSICAL_ADDRESS HighAddress, IN PHYSICAL_ADDRESS SkipBytes, IN SIZE_T TotalBytes)
Definition: mdlsup.c:489
VOID NTAPI MmBuildMdlForNonPagedPool(IN PMDL Mdl)
Definition: mdlsup.c:424
PVOID NTAPI MmMapLockedPagesSpecifyCache(IN PMDL Mdl, IN KPROCESSOR_MODE AccessMode, IN MEMORY_CACHING_TYPE CacheType, IN PVOID BaseAddress, IN ULONG BugCheckOnFailure, IN ULONG Priority)
Definition: mdlsup.c:660
VOID NTAPI MmUnmapLockedPages(IN PVOID BaseAddress, IN PMDL Mdl)
Definition: mdlsup.c:837
#define ExFreePoolWithTag(_P, _T)
Definition: module.h:1109
@ NormalPagePriority
Definition: imports.h:54
#define KernelMode
Definition: asm.h:38
_In_ PVOID _Out_opt_ BOOLEAN _Out_opt_ PPFN_NUMBER Page
Definition: mm.h:1305
ULONG * PPFN_NUMBER
Definition: ke.h:9
#define _WIN32_WINNT_WIN8
Definition: sdkddkver.h:29
uint32_t ULONG
Definition: typedefs.h:59
uint64_t ULONGLONG
Definition: typedefs.h:67
LONGLONG QuadPart
Definition: typedefs.h:114
_In_ WDFDEVICE _In_ PVOID _In_opt_ PMDL Mdl
#define MmGetMdlByteCount(_Mdl)
_Must_inspect_result_ _In_ PHYSICAL_ADDRESS HighAddress
Definition: mmfuncs.h:226
#define MmGetMdlVirtualAddress(_Mdl)
#define MmGetMdlPfnArray(_Mdl)
#define ADDRESS_AND_SIZE_TO_SPAN_PAGES(_Va, _Size)
_Must_inspect_result_ _In_ PHYSICAL_ADDRESS _In_ PHYSICAL_ADDRESS SkipBytes
Definition: mmfuncs.h:227
@ MmCached
Definition: mmtypes.h:130
#define MDL_PAGES_LOCKED
Definition: mmtypes.h:19
#define MDL_SOURCE_IS_NONPAGED_POOL
Definition: mmtypes.h:20
#define MDL_MAPPED_TO_SYSTEM_VA
Definition: mmtypes.h:18