summaryrefslogtreecommitdiffstats
path: root/base/directory_watcher_win.cc
blob: f9782779ba4fbdb01bdf997f6123baa76a3cd686 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
// Copyright (c) 2006-2008 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

#include "base/directory_watcher.h"

#include "base/file_path.h"
#include "base/object_watcher.h"

// Private implementation class implementing the behavior of DirectoryWatcher.
class DirectoryWatcher::Impl : public base::RefCounted<DirectoryWatcher::Impl>,
                               public base::ObjectWatcher::Delegate {
 public:
  Impl(DirectoryWatcher::Delegate* delegate)
      : delegate_(delegate), handle_(INVALID_HANDLE_VALUE) {}
  ~Impl();

  // Register interest in any changes in |path|.
  // Returns false on error.
  bool Watch(const FilePath& path);

  // Callback from MessageLoopForIO.
  virtual void OnObjectSignaled(HANDLE object);

 private:
  // Delegate to notify upon changes.
  DirectoryWatcher::Delegate* delegate_;
  // Path we're watching (passed to delegate).
  FilePath path_;
  // Handle for FindFirstChangeNotification.
  HANDLE handle_;
  // ObjectWatcher to watch handle_ for events.
  base::ObjectWatcher watcher_;
};

DirectoryWatcher::Impl::~Impl() {
  if (handle_ != INVALID_HANDLE_VALUE) {
    watcher_.StopWatching();
    FindCloseChangeNotification(handle_);
  }
}

bool DirectoryWatcher::Impl::Watch(const FilePath& path) {
  DCHECK(path_.value().empty());  // Can only watch one path.

  // NOTE: If you want to change this code to *not* watch subdirectories, have a
  // look at http://code.google.com/p/chromium/issues/detail?id=5072 first.
  handle_ = FindFirstChangeNotification(
      path.value().c_str(),
      TRUE,  // Watch subtree.
      FILE_NOTIFY_CHANGE_FILE_NAME | FILE_NOTIFY_CHANGE_SIZE |
      FILE_NOTIFY_CHANGE_LAST_WRITE);
  if (handle_ == INVALID_HANDLE_VALUE)
    return false;

  path_ = path;
  watcher_.StartWatching(handle_, this);

  return true;
}

void DirectoryWatcher::Impl::OnObjectSignaled(HANDLE object) {
  DCHECK(object == handle_);
  // Make sure we stay alive through the body of this function.
  scoped_refptr<DirectoryWatcher::Impl> keep_alive(this);

  delegate_->OnDirectoryChanged(path_);

  // Register for more notifications on file change.
  BOOL ok = FindNextChangeNotification(object);
  DCHECK(ok);
  watcher_.StartWatching(object, this);
}

DirectoryWatcher::DirectoryWatcher() {
}

DirectoryWatcher::~DirectoryWatcher() {
  // Declared in .cc file for access to ~DirectoryWatcher::Impl.
}

bool DirectoryWatcher::Watch(const FilePath& path,
                             Delegate* delegate) {
  impl_ = new DirectoryWatcher::Impl(delegate);
  return impl_->Watch(path);
}