2 * File change notification tests
4 * Copyright 2006 Mike McCormack for CodeWeavers
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
21 #include "ntdll_test.h"
23 static NTSTATUS (WINAPI
*pNtNotifyChangeDirectoryFile
)(
24 HANDLE
,HANDLE
,PIO_APC_ROUTINE
,PVOID
,
25 PIO_STATUS_BLOCK
,PVOID
,ULONG
,ULONG
,BOOLEAN
);
27 static NTSTATUS (WINAPI
*pNtCancelIoFile
)(HANDLE
,PIO_STATUS_BLOCK
);
30 static void test_ntncdf(void)
35 DWORD fflags
, filter
= 0;
37 WCHAR path
[MAX_PATH
], subdir
[MAX_PATH
];
38 static const WCHAR szBoo
[] = { '\\','b','o','o',0 };
39 static const WCHAR szHoo
[] = { '\\','h','o','o',0 };
40 PFILE_NOTIFY_INFORMATION pfni
;
42 r
= GetTempPathW( MAX_PATH
, path
);
43 ok( r
!= 0, "temp path failed\n");
47 lstrcatW( path
, szBoo
);
48 lstrcpyW( subdir
, path
);
49 lstrcatW( subdir
, szHoo
);
51 RemoveDirectoryW( subdir
);
52 RemoveDirectoryW( path
);
54 r
= CreateDirectoryW(path
, NULL
);
55 ok( r
== TRUE
, "failed to create directory\n");
57 r
= pNtNotifyChangeDirectoryFile(NULL
,NULL
,NULL
,NULL
,NULL
,NULL
,0,0,0);
58 ok(r
==STATUS_ACCESS_VIOLATION
, "should return access violation\n");
60 fflags
= FILE_FLAG_BACKUP_SEMANTICS
| FILE_FLAG_OVERLAPPED
;
61 hdir
= CreateFileW(path
, GENERIC_READ
|SYNCHRONIZE
, FILE_SHARE_READ
, NULL
,
62 OPEN_EXISTING
, fflags
, NULL
);
63 ok( hdir
!= INVALID_HANDLE_VALUE
, "failed to open directory\n");
65 hEvent
= CreateEventA( NULL
, 0, 0, NULL
);
67 r
= pNtNotifyChangeDirectoryFile(hdir
,NULL
,NULL
,NULL
,&iosb
,NULL
,0,0,0);
68 ok(r
==STATUS_INVALID_PARAMETER
, "should return invalid parameter\n");
70 r
= pNtNotifyChangeDirectoryFile(hdir
,hEvent
,NULL
,NULL
,&iosb
,NULL
,0,0,0);
71 ok(r
==STATUS_INVALID_PARAMETER
, "should return invalid parameter\n");
73 filter
= FILE_NOTIFY_CHANGE_FILE_NAME
;
74 filter
|= FILE_NOTIFY_CHANGE_DIR_NAME
;
75 filter
|= FILE_NOTIFY_CHANGE_ATTRIBUTES
;
76 filter
|= FILE_NOTIFY_CHANGE_SIZE
;
77 filter
|= FILE_NOTIFY_CHANGE_LAST_WRITE
;
78 filter
|= FILE_NOTIFY_CHANGE_LAST_ACCESS
;
79 filter
|= FILE_NOTIFY_CHANGE_CREATION
;
80 filter
|= FILE_NOTIFY_CHANGE_SECURITY
;
84 r
= pNtNotifyChangeDirectoryFile(hdir
,hEvent
,NULL
,NULL
,&iosb
,buffer
,sizeof buffer
,-1,0);
85 ok(r
==STATUS_INVALID_PARAMETER
, "should return invalid parameter\n");
87 ok( U(iosb
).Status
== 1, "information wrong\n");
88 ok( iosb
.Information
== 1, "information wrong\n");
92 r
= pNtNotifyChangeDirectoryFile(hdir
,hEvent
,NULL
,NULL
,&iosb
,buffer
,sizeof buffer
,filter
,0);
93 ok(r
==STATUS_PENDING
, "should return status pending\n");
95 r
= WaitForSingleObject( hEvent
, 100 );
96 ok( r
== STATUS_TIMEOUT
, "should timeout\n" );
98 r
= WaitForSingleObject( hdir
, 100 );
99 ok( r
== STATUS_TIMEOUT
, "should timeout\n" );
101 r
= CreateDirectoryW( subdir
, NULL
);
102 ok( r
== TRUE
, "failed to create directory\n");
104 r
= WaitForSingleObject( hdir
, 100 );
105 ok( r
== STATUS_TIMEOUT
, "should timeout\n" );
107 r
= WaitForSingleObject( hEvent
, 100 );
108 ok( r
== WAIT_OBJECT_0
, "event should be ready\n" );
110 ok( U(iosb
).Status
== STATUS_SUCCESS
, "information wrong\n");
111 ok( iosb
.Information
== 0x12, "information wrong\n");
113 pfni
= (PFILE_NOTIFY_INFORMATION
) buffer
;
114 ok( pfni
->NextEntryOffset
== 0, "offset wrong\n" );
115 ok( pfni
->Action
== FILE_ACTION_ADDED
, "action wrong\n" );
116 ok( pfni
->FileNameLength
== 6, "len wrong\n" );
117 ok( !memcmp(pfni
->FileName
,&szHoo
[1],6), "name wrong\n" );
119 r
= pNtNotifyChangeDirectoryFile(hdir
,0,NULL
,NULL
,&iosb
,buffer
,sizeof buffer
,0,0);
120 ok(r
==STATUS_INVALID_PARAMETER
, "should return invalid parameter\n");
122 r
= pNtNotifyChangeDirectoryFile(hdir
,hEvent
,NULL
,NULL
,&iosb
,buffer
,sizeof buffer
,0,0);
123 ok(r
==STATUS_INVALID_PARAMETER
, "should return invalid parameter\n");
125 filter
= FILE_NOTIFY_CHANGE_SIZE
;
128 iosb
.Information
= 1;
129 r
= pNtNotifyChangeDirectoryFile(hdir
,0,NULL
,NULL
,&iosb
,NULL
,0,filter
,0);
130 ok(r
==STATUS_PENDING
, "should status pending\n");
132 ok( U(iosb
).Status
== 1, "information wrong\n");
133 ok( iosb
.Information
== 1, "information wrong\n");
135 r
= WaitForSingleObject( hdir
, 0 );
136 ok( r
== STATUS_TIMEOUT
, "should timeout\n" );
138 r
= RemoveDirectoryW( subdir
);
139 ok( r
== TRUE
, "failed to remove directory\n");
141 r
= WaitForSingleObject( hdir
, 100 );
142 ok( r
== WAIT_OBJECT_0
, "should be ready\n" );
144 r
= WaitForSingleObject( hdir
, 100 );
145 ok( r
== WAIT_OBJECT_0
, "should be ready\n" );
147 ok( U(iosb
).Status
== STATUS_NOTIFY_ENUM_DIR
, "information wrong\n");
148 ok( iosb
.Information
== 0, "information wrong\n");
153 r
= RemoveDirectoryW( path
);
154 ok( r
== TRUE
, "failed to remove directory\n");
158 static void test_ntncdf_async(void)
163 DWORD fflags
, filter
= 0;
164 IO_STATUS_BLOCK iosb
, iosb2
;
165 WCHAR path
[MAX_PATH
], subdir
[MAX_PATH
];
166 static const WCHAR szBoo
[] = { '\\','b','o','o',0 };
167 static const WCHAR szHoo
[] = { '\\','h','o','o',0 };
168 PFILE_NOTIFY_INFORMATION pfni
;
170 r
= GetTempPathW( MAX_PATH
, path
);
171 ok( r
!= 0, "temp path failed\n");
175 lstrcatW( path
, szBoo
);
176 lstrcpyW( subdir
, path
);
177 lstrcatW( subdir
, szHoo
);
179 RemoveDirectoryW( subdir
);
180 RemoveDirectoryW( path
);
182 r
= CreateDirectoryW(path
, NULL
);
183 ok( r
== TRUE
, "failed to create directory\n");
185 r
= pNtNotifyChangeDirectoryFile(NULL
,NULL
,NULL
,NULL
,NULL
,NULL
,0,0,0);
186 ok(r
==STATUS_ACCESS_VIOLATION
, "should return access violation\n");
188 fflags
= FILE_FLAG_BACKUP_SEMANTICS
| FILE_FLAG_OVERLAPPED
;
189 hdir
= CreateFileW(path
, GENERIC_READ
|SYNCHRONIZE
, FILE_SHARE_READ
, NULL
,
190 OPEN_EXISTING
, fflags
, NULL
);
191 ok( hdir
!= INVALID_HANDLE_VALUE
, "failed to open directory\n");
193 hEvent
= CreateEventA( NULL
, 0, 0, NULL
);
195 filter
= FILE_NOTIFY_CHANGE_FILE_NAME
;
196 filter
|= FILE_NOTIFY_CHANGE_DIR_NAME
;
197 filter
|= FILE_NOTIFY_CHANGE_ATTRIBUTES
;
198 filter
|= FILE_NOTIFY_CHANGE_SIZE
;
199 filter
|= FILE_NOTIFY_CHANGE_LAST_WRITE
;
200 filter
|= FILE_NOTIFY_CHANGE_LAST_ACCESS
;
201 filter
|= FILE_NOTIFY_CHANGE_CREATION
;
202 filter
|= FILE_NOTIFY_CHANGE_SECURITY
;
205 U(iosb
).Status
= 0x01234567;
206 iosb
.Information
= 0x12345678;
207 r
= pNtNotifyChangeDirectoryFile(hdir
,0,NULL
,NULL
,&iosb
,buffer
,sizeof buffer
,filter
,0);
208 ok(r
==STATUS_PENDING
, "should status pending\n");
209 ok(U(iosb
).Status
== 0x01234567, "status set too soon\n");
210 ok(iosb
.Information
== 0x12345678, "info set too soon\n");
212 r
= CreateDirectoryW( subdir
, NULL
);
213 ok( r
== TRUE
, "failed to create directory\n");
215 r
= WaitForSingleObject( hdir
, 100 );
216 ok( r
== WAIT_OBJECT_0
, "should be ready\n" );
218 ok(U(iosb
).Status
== STATUS_SUCCESS
, "status not successful\n");
219 ok(iosb
.Information
== 0x12, "info not set\n");
221 pfni
= (PFILE_NOTIFY_INFORMATION
) buffer
;
222 ok( pfni
->NextEntryOffset
== 0, "offset wrong\n" );
223 ok( pfni
->Action
== FILE_ACTION_ADDED
, "action wrong\n" );
224 ok( pfni
->FileNameLength
== 6, "len wrong\n" );
225 ok( !memcmp(pfni
->FileName
,&szHoo
[1],6), "name wrong\n" );
227 r
= pNtNotifyChangeDirectoryFile(hdir
,0,NULL
,NULL
,&iosb
,buffer
,sizeof buffer
,filter
,0);
228 ok(r
==STATUS_PENDING
, "should status pending\n");
230 r
= RemoveDirectoryW( subdir
);
231 ok( r
== TRUE
, "failed to remove directory\n");
233 r
= WaitForSingleObject( hdir
, 0 );
234 ok( r
== WAIT_OBJECT_0
, "should be ready\n" );
236 ok(U(iosb
).Status
== STATUS_SUCCESS
, "status not successful\n");
237 ok(iosb
.Information
== 0x12, "info not set\n");
239 ok( pfni
->NextEntryOffset
== 0, "offset wrong\n" );
240 ok( pfni
->Action
== FILE_ACTION_REMOVED
, "action wrong\n" );
241 ok( pfni
->FileNameLength
== 6, "len wrong\n" );
242 ok( !memcmp(pfni
->FileName
,&szHoo
[1],6), "name wrong\n" );
246 iosb
.Information
= 0;
248 r
= pNtNotifyChangeDirectoryFile(hdir
,0,NULL
,NULL
,&iosb
,NULL
,0,filter
,0);
249 ok(r
==STATUS_PENDING
, "should status pending\n");
251 r
= CreateDirectoryW( subdir
, NULL
);
252 ok( r
== TRUE
, "failed to create directory\n");
254 r
= WaitForSingleObject( hdir
, 0 );
255 ok( r
== WAIT_OBJECT_0
, "should be ready\n" );
257 ok(U(iosb
).Status
== STATUS_NOTIFY_ENUM_DIR
, "status not successful\n");
258 ok(iosb
.Information
== 0, "info not set\n");
261 iosb
.Information
= 0;
263 r
= pNtNotifyChangeDirectoryFile(hdir
,hEvent
,NULL
,NULL
,&iosb
,buffer
,sizeof buffer
,filter
,0);
264 ok(r
==STATUS_PENDING
, "should status pending\n");
266 r
= RemoveDirectoryW( subdir
);
267 ok( r
== TRUE
, "failed to remove directory\n");
269 r
= WaitForSingleObject( hEvent
, 0 );
270 ok( r
== WAIT_OBJECT_0
, "should be ready\n" );
272 ok(U(iosb
).Status
== STATUS_SUCCESS
, "status not successful\n");
273 ok(iosb
.Information
== 0x12, "info not set\n");
276 U(iosb
).Status
= 0x01234567;
277 iosb
.Information
= 0x12345678;
278 r
= pNtNotifyChangeDirectoryFile(hdir
,0,NULL
,NULL
,&iosb
,buffer
,sizeof buffer
,filter
,0);
279 ok(r
==STATUS_PENDING
, "should status pending\n");
281 U(iosb2
).Status
= 0x01234567;
282 iosb2
.Information
= 0x12345678;
283 r
= pNtNotifyChangeDirectoryFile(hdir
,0,NULL
,NULL
,&iosb2
,buffer
,sizeof buffer
,filter
,0);
284 ok(r
==STATUS_PENDING
, "should status pending\n");
286 ok(U(iosb
).Status
== 0x01234567, "status set too soon\n");
287 ok(iosb
.Information
== 0x12345678, "info set too soon\n");
289 r
= pNtCancelIoFile(hdir
, &iosb
);
290 ok( r
== STATUS_SUCCESS
, "cancel failed\n");
294 ok(U(iosb
).Status
== STATUS_SUCCESS
, "status wrong\n");
295 ok(U(iosb2
).Status
== STATUS_CANCELLED
, "status wrong %x\n",U(iosb2
).Status
);
297 ok(iosb
.Information
== 0, "info wrong\n");
298 ok(iosb2
.Information
== 0, "info wrong\n");
300 r
= RemoveDirectoryW( path
);
301 ok( r
== TRUE
, "failed to remove directory\n");
308 HMODULE hntdll
= GetModuleHandleA("ntdll");
311 win_skip("not running on NT, skipping test\n");
315 pNtNotifyChangeDirectoryFile
= (void *)GetProcAddress(hntdll
, "NtNotifyChangeDirectoryFile");
316 pNtCancelIoFile
= (void *)GetProcAddress(hntdll
, "NtCancelIoFile");
318 if (!pNtNotifyChangeDirectoryFile
|| !pNtCancelIoFile
)
320 win_skip("missing functions, skipping test\n");