change.c 6.5 KB
Newer Older
Alexandre Julliard's avatar
Alexandre Julliard committed
1 2 3 4
/*
 * Win32 file change notification functions
 *
 * Copyright 1998 Ulrich Weigand
5 6 7 8 9 10 11 12 13 14 15 16 17
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2.1 of the License, or (at your option) any later version.
 *
 * This library is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * 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
Alexandre Julliard's avatar
Alexandre Julliard committed
19 20
 */

21 22
#include "config.h"

23
#include <stdarg.h>
Alexandre Julliard's avatar
Alexandre Julliard committed
24 25
#include <stdlib.h>
#include <string.h>
26

27 28
#include "ntstatus.h"
#define WIN32_NO_STATUS
29 30
#define NONAMELESSUNION
#define NONAMELESSSTRUCT
31
#include "windef.h"
Alexandre Julliard's avatar
Alexandre Julliard committed
32 33
#include "winbase.h"
#include "winerror.h"
34
#include "winternl.h"
35
#include "kernel_private.h"
36
#include "wine/debug.h"
Alexandre Julliard's avatar
Alexandre Julliard committed
37

38
WINE_DEFAULT_DEBUG_CHANNEL(file);
Alexandre Julliard's avatar
Alexandre Julliard committed
39 40

/****************************************************************************
41
 *		FindFirstChangeNotificationA (KERNEL32.@)
Alexandre Julliard's avatar
Alexandre Julliard committed
42
 */
43
HANDLE WINAPI FindFirstChangeNotificationA( LPCSTR lpPathName, BOOL bWatchSubtree,
44
                                            DWORD dwNotifyFilter )
Alexandre Julliard's avatar
Alexandre Julliard committed
45
{
46
    WCHAR *pathW;
47

48 49
    if (!(pathW = FILE_name_AtoW( lpPathName, FALSE ))) return INVALID_HANDLE_VALUE;
    return FindFirstChangeNotificationW( pathW, bWatchSubtree, dwNotifyFilter );
Alexandre Julliard's avatar
Alexandre Julliard committed
50 51
}

52 53 54 55 56 57 58
/*
 * NtNotifyChangeDirectoryFile may write back to the IO_STATUS_BLOCK
 * asynchronously.  We don't care about the contents, but it can't
 * be placed on the stack since it will go out of scope when we return.
 */
static IO_STATUS_BLOCK FindFirstChange_iosb;

Alexandre Julliard's avatar
Alexandre Julliard committed
59
/****************************************************************************
60
 *		FindFirstChangeNotificationW (KERNEL32.@)
Alexandre Julliard's avatar
Alexandre Julliard committed
61
 */
62 63
HANDLE WINAPI FindFirstChangeNotificationW( LPCWSTR lpPathName, BOOL bWatchSubtree,
                                            DWORD dwNotifyFilter)
Alexandre Julliard's avatar
Alexandre Julliard committed
64
{
65 66 67
    UNICODE_STRING nt_name;
    OBJECT_ATTRIBUTES attr;
    NTSTATUS status;
68
    HANDLE handle = INVALID_HANDLE_VALUE;
69

70
    TRACE( "%s %d %x\n", debugstr_w(lpPathName), bWatchSubtree, dwNotifyFilter );
71

72 73 74
    if (!RtlDosPathNameToNtPathName_U( lpPathName, &nt_name, NULL, NULL ))
    {
        SetLastError( ERROR_PATH_NOT_FOUND );
75
        return handle;
76 77 78 79 80 81 82 83 84
    }

    attr.Length = sizeof(attr);
    attr.RootDirectory = 0;
    attr.Attributes = OBJ_CASE_INSENSITIVE;
    attr.ObjectName = &nt_name;
    attr.SecurityDescriptor = NULL;
    attr.SecurityQualityOfService = NULL;

85 86
    status = NtOpenFile( &handle, FILE_LIST_DIRECTORY | SYNCHRONIZE,
                         &attr, &FindFirstChange_iosb,
87
                         FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE,
88 89 90 91 92 93
                         FILE_DIRECTORY_FILE | FILE_SYNCHRONOUS_IO_NONALERT );
    RtlFreeUnicodeString( &nt_name );

    if (status != STATUS_SUCCESS)
    {
        SetLastError( RtlNtStatusToDosError(status) );
94
        return INVALID_HANDLE_VALUE;
95
    }
96

97 98
    status = NtNotifyChangeDirectoryFile( handle, NULL, NULL, NULL,
                                          &FindFirstChange_iosb,
99 100
                                          NULL, 0, dwNotifyFilter, bWatchSubtree );
    if (status != STATUS_PENDING)
101
    {
102 103 104
        NtClose( handle );
        SetLastError( RtlNtStatusToDosError(status) );
        return INVALID_HANDLE_VALUE;
105
    }
106
    return handle;
Alexandre Julliard's avatar
Alexandre Julliard committed
107 108 109
}

/****************************************************************************
110
 *		FindNextChangeNotification (KERNEL32.@)
Alexandre Julliard's avatar
Alexandre Julliard committed
111
 */
112
BOOL WINAPI FindNextChangeNotification( HANDLE handle )
Alexandre Julliard's avatar
Alexandre Julliard committed
113
{
114
    NTSTATUS status;
115 116 117

    TRACE("%p\n",handle);

118 119
    status = NtNotifyChangeDirectoryFile( handle, NULL, NULL, NULL,
                                          &FindFirstChange_iosb,
120 121
                                          NULL, 0, FILE_NOTIFY_CHANGE_SIZE, 0 );
    if (status != STATUS_PENDING)
122
    {
123 124
        SetLastError( RtlNtStatusToDosError(status) );
        return FALSE;
125
    }
126
    return TRUE;
Alexandre Julliard's avatar
Alexandre Julliard committed
127 128 129
}

/****************************************************************************
130
 *		FindCloseChangeNotification (KERNEL32.@)
Alexandre Julliard's avatar
Alexandre Julliard committed
131
 */
132
BOOL WINAPI FindCloseChangeNotification( HANDLE handle )
Alexandre Julliard's avatar
Alexandre Julliard committed
133 134 135
{
    return CloseHandle( handle );
}
136

137 138 139 140 141 142
static void WINAPI invoke_completion(LPVOID ctx, IO_STATUS_BLOCK *ios, ULONG res)
{
    LPOVERLAPPED_COMPLETION_ROUTINE completion = ctx;
    completion(ios->u.Status, ios->Information, (LPOVERLAPPED)ios);
}

143 144 145 146 147 148 149 150 151 152 153 154 155
/****************************************************************************
 *		ReadDirectoryChangesW (KERNEL32.@)
 *
 * NOTES
 *
 *  The filter is remember from the first run and ignored on successive runs.
 *
 *  If there's no output buffer on the first run, it's ignored successive runs
 *   and STATUS_NOTIFY_ENUM_DIRECTORY is returned with an empty buffer.
 *
 *  If a NULL overlapped->hEvent is passed, the directory handle is used
 *   for signalling.
 */
156 157 158 159
BOOL WINAPI ReadDirectoryChangesW( HANDLE handle, LPVOID buffer, DWORD len, BOOL subtree,
                                   DWORD filter, LPDWORD returned, LPOVERLAPPED overlapped,
                                   LPOVERLAPPED_COMPLETION_ROUTINE completion )
{
160 161
    OVERLAPPED ov, *pov;
    IO_STATUS_BLOCK *ios;
162 163
    NTSTATUS status;
    BOOL ret = TRUE;
164
    LPVOID cvalue = NULL;
165

166
    TRACE("%p %p %08x %d %08x %p %p %p\n", handle, buffer, len, subtree, filter,
167 168
           returned, overlapped, completion );

169 170 171 172 173 174
    if (!overlapped)
    {
        memset( &ov, 0, sizeof ov );
        ov.hEvent = CreateEventW( NULL, 0, 0, NULL );
        pov = &ov;
    }
175
    else
176
    {
177
        pov = overlapped;
178 179
        if(completion) cvalue = completion;
        else if (((ULONG_PTR)overlapped->hEvent & 1) == 0) cvalue = overlapped;
180
    }
181

182
    ios = (PIO_STATUS_BLOCK) pov;
183
    ios->u.Status = STATUS_PENDING;
184

185 186 187
    status = NtNotifyChangeDirectoryFile( handle, completion && overlapped ? NULL : pov->hEvent,
            completion && overlapped ? invoke_completion : NULL,
            cvalue, ios, buffer, len, filter, subtree );
188 189 190 191 192 193 194 195 196
    if (status == STATUS_PENDING)
    {
        if (overlapped)
            return TRUE;

        WaitForSingleObjectEx( ov.hEvent, INFINITE, TRUE );
        CloseHandle( ov.hEvent );
        if (returned)
            *returned = ios->Information;
197
        status = ios->u.Status;
198 199 200
    }

    if (status != STATUS_SUCCESS)
201 202 203 204 205 206
    {
        SetLastError( RtlNtStatusToDosError(status) );
        ret = FALSE;
    }

    return ret;
207
}