kernel: Add some more tests for FindFirstChangeNotification.
[wine] / dlls / kernel / change.c
1 /*
2  * Win32 file change notification functions
3  *
4  * Copyright 1998 Ulrich Weigand
5  *
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.
10  *
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.
15  *
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., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
19  */
20
21 #include "config.h"
22
23 #include <stdarg.h>
24 #include <stdlib.h>
25 #include <string.h>
26
27 #include "ntstatus.h"
28 #define WIN32_NO_STATUS
29 #include "windef.h"
30 #include "winbase.h"
31 #include "winerror.h"
32 #include "winternl.h"
33 #include "kernel_private.h"
34 #include "wine/debug.h"
35
36 WINE_DEFAULT_DEBUG_CHANNEL(file);
37
38 /****************************************************************************
39  *              FindFirstChangeNotificationA (KERNEL32.@)
40  */
41 HANDLE WINAPI FindFirstChangeNotificationA( LPCSTR lpPathName, BOOL bWatchSubtree,
42                                             DWORD dwNotifyFilter )
43 {
44     WCHAR *pathW;
45
46     if (!(pathW = FILE_name_AtoW( lpPathName, FALSE ))) return INVALID_HANDLE_VALUE;
47     return FindFirstChangeNotificationW( pathW, bWatchSubtree, dwNotifyFilter );
48 }
49
50 /****************************************************************************
51  *              FindFirstChangeNotificationW (KERNEL32.@)
52  */
53 HANDLE WINAPI FindFirstChangeNotificationW( LPCWSTR lpPathName, BOOL bWatchSubtree,
54                                             DWORD dwNotifyFilter)
55 {
56     UNICODE_STRING nt_name;
57     OBJECT_ATTRIBUTES attr;
58     IO_STATUS_BLOCK io;
59     NTSTATUS status;
60     HANDLE handle = INVALID_HANDLE_VALUE;
61
62     TRACE( "%s %d %lx\n", debugstr_w(lpPathName), bWatchSubtree, dwNotifyFilter );
63
64     if (!RtlDosPathNameToNtPathName_U( lpPathName, &nt_name, NULL, NULL ))
65     {
66         SetLastError( ERROR_PATH_NOT_FOUND );
67         return handle;
68     }
69
70     attr.Length = sizeof(attr);
71     attr.RootDirectory = 0;
72     attr.Attributes = OBJ_CASE_INSENSITIVE;
73     attr.ObjectName = &nt_name;
74     attr.SecurityDescriptor = NULL;
75     attr.SecurityQualityOfService = NULL;
76
77     status = NtOpenFile( &handle, SYNCHRONIZE, &attr, &io,
78                          FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE,
79                          FILE_DIRECTORY_FILE | FILE_SYNCHRONOUS_IO_NONALERT );
80     RtlFreeUnicodeString( &nt_name );
81
82     if (status != STATUS_SUCCESS)
83     {
84         SetLastError( RtlNtStatusToDosError(status) );
85         return INVALID_HANDLE_VALUE;
86     }
87
88     status = NtNotifyChangeDirectoryFile( handle, NULL, NULL, NULL, &io,
89                                           NULL, 0, dwNotifyFilter, bWatchSubtree );
90     if (status != STATUS_PENDING)
91     {
92         NtClose( handle );
93         SetLastError( RtlNtStatusToDosError(status) );
94         return INVALID_HANDLE_VALUE;
95     }
96     return handle;
97 }
98
99 /****************************************************************************
100  *              FindNextChangeNotification (KERNEL32.@)
101  */
102 BOOL WINAPI FindNextChangeNotification( HANDLE handle )
103 {
104     IO_STATUS_BLOCK io;
105     NTSTATUS status;
106
107     TRACE("%p\n",handle);
108
109     status = NtNotifyChangeDirectoryFile( handle, NULL, NULL, NULL, &io,
110                                           NULL, 0, FILE_NOTIFY_CHANGE_SIZE, 0 );
111     if (status != STATUS_PENDING)
112     {
113         SetLastError( RtlNtStatusToDosError(status) );
114         return FALSE;
115     }
116     return TRUE;
117 }
118
119 /****************************************************************************
120  *              FindCloseChangeNotification (KERNEL32.@)
121  */
122 BOOL WINAPI FindCloseChangeNotification( HANDLE handle )
123 {
124     return CloseHandle( handle );
125 }
126
127 BOOL WINAPI ReadDirectoryChangesW( HANDLE handle, LPVOID buffer, DWORD len, BOOL subtree,
128                                    DWORD filter, LPDWORD returned, LPOVERLAPPED overlapped,
129                                    LPOVERLAPPED_COMPLETION_ROUTINE completion )
130 {
131     IO_STATUS_BLOCK io;
132     NTSTATUS status;
133     BOOL ret = TRUE;
134     HANDLE event;
135
136     TRACE("%p %p %08lx %d %08lx %p %p %p\n", handle, buffer, len, subtree, filter,
137            returned, overlapped, completion );
138
139     if (overlapped)
140         event = overlapped->hEvent;
141     else
142         event = CreateEventW( NULL, 0, 0, NULL );
143
144     status = NtNotifyChangeDirectoryFile( handle, event, NULL, NULL,
145                                               &io, buffer, len, filter, subtree );
146     if (status != STATUS_PENDING)
147     {
148         SetLastError( RtlNtStatusToDosError(status) );
149         ret = FALSE;
150     }
151     else if (!overlapped)
152         WaitForSingleObject( event, INFINITE );
153
154     if (!overlapped)
155         CloseHandle( event );
156
157     return ret;
158 }