summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authordavidben@chromium.org <davidben@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2010-08-27 01:29:28 +0000
committerdavidben@chromium.org <davidben@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2010-08-27 01:29:28 +0000
commit53f4826c9fe6bea1718a499dc686f26dbf24ab50 (patch)
treeb2d1f444aa73e927ff215e7dc78e851bc213a056
parent34d3608cd3c631811eef51bc5cdd8ce2a2670076 (diff)
downloadchromium_src-53f4826c9fe6bea1718a499dc686f26dbf24ab50.zip
chromium_src-53f4826c9fe6bea1718a499dc686f26dbf24ab50.tar.gz
chromium_src-53f4826c9fe6bea1718a499dc686f26dbf24ab50.tar.bz2
Move the SingletonSocket to a temporary directory
This is to workaround problems on certain network filesystems (notably AFS) which do not support Unix domain sockets. We move the sockets into a temporary folder and symlink. To avoid the possibility of a dangling link to a missing (and later intercepted) remote directory, we create and check cookie files and rely on the stickiness of /tmp/ to avoid a race condition in the check. R=mattm BUG=44606 TEST=ProcessSingletonLinuxTest.* Review URL: http://codereview.chromium.org/2838034 git-svn-id: svn://svn.chromium.org/chrome/trunk/src@57623 0039d316-1c4b-4281-b951-d872f2087c98
-rw-r--r--base/scoped_temp_dir.cc9
-rw-r--r--base/scoped_temp_dir.h5
-rw-r--r--chrome/browser/process_singleton.h9
-rw-r--r--chrome/browser/process_singleton_linux.cc235
-rw-r--r--chrome/browser/process_singleton_linux_uitest.cc56
-rw-r--r--chrome/common/chrome_constants.cc3
-rw-r--r--chrome/common/chrome_constants.h1
7 files changed, 253 insertions, 65 deletions
diff --git a/base/scoped_temp_dir.cc b/base/scoped_temp_dir.cc
index 958dcbc..a510ddf 100644
--- a/base/scoped_temp_dir.cc
+++ b/base/scoped_temp_dir.cc
@@ -11,8 +11,7 @@ ScopedTempDir::ScopedTempDir() {
}
ScopedTempDir::~ScopedTempDir() {
- if (!path_.empty() && !file_util::Delete(path_, true))
- LOG(ERROR) << "ScopedTempDir unable to delete " << path_.value();
+ Delete();
}
bool ScopedTempDir::CreateUniqueTempDir() {
@@ -50,6 +49,12 @@ bool ScopedTempDir::Set(const FilePath& path) {
return true;
}
+void ScopedTempDir::Delete() {
+ if (!path_.empty() && !file_util::Delete(path_, true))
+ LOG(ERROR) << "ScopedTempDir unable to delete " << path_.value();
+ path_.clear();
+}
+
FilePath ScopedTempDir::Take() {
FilePath ret = path_;
path_ = FilePath();
diff --git a/base/scoped_temp_dir.h b/base/scoped_temp_dir.h
index 702c2bc..66d52f6 100644
--- a/base/scoped_temp_dir.h
+++ b/base/scoped_temp_dir.h
@@ -19,7 +19,7 @@ class ScopedTempDir {
// No directory is owned/created initially.
ScopedTempDir();
- // Recursively delete path_
+ // Recursively delete path.
~ScopedTempDir();
// Creates a unique directory in TempPath, and takes ownership of it.
@@ -33,6 +33,9 @@ class ScopedTempDir {
// Don't call multiple times unless Take() has been called first.
bool Set(const FilePath& path);
+ // Deletes the temporary directory wrapped by this object.
+ void Delete();
+
// Caller takes ownership of the temporary directory so it won't be destroyed
// when this object goes out of scope.
FilePath Take();
diff --git a/chrome/browser/process_singleton.h b/chrome/browser/process_singleton.h
index 19e40a6..57dd446 100644
--- a/chrome/browser/process_singleton.h
+++ b/chrome/browser/process_singleton.h
@@ -20,6 +20,9 @@
#include "base/non_thread_safe.h"
#include "base/ref_counted.h"
#include "gfx/native_widget_types.h"
+#if defined(USE_X11)
+#include "base/scoped_temp_dir.h"
+#endif
class CommandLine;
class FilePath;
@@ -139,6 +142,12 @@ class ProcessSingleton : public NonThreadSafe {
// Path in file system to the lock.
FilePath lock_path_;
+ // Path in file system to the cookie file.
+ FilePath cookie_path_;
+
+ // Temporary directory to hold the socket.
+ ScopedTempDir socket_dir_;
+
// Helper class for linux specific messages. LinuxWatcher is ref counted
// because it posts messages between threads.
class LinuxWatcher;
diff --git a/chrome/browser/process_singleton_linux.cc b/chrome/browser/process_singleton_linux.cc
index 8a30e3e..d064987 100644
--- a/chrome/browser/process_singleton_linux.cc
+++ b/chrome/browser/process_singleton_linux.cc
@@ -8,6 +8,16 @@
// directory and second process command line flags. The second process then
// exits.
//
+// Because many networked filesystem implementations do not support unix domain
+// sockets, we create the socket in a temporary directory and create a symlink
+// in the profile. This temporary directory is no longer bound to the profile,
+// and may disappear across a reboot or login to a separate session. To bind
+// them, we store a unique cookie in the profile directory, which must also be
+// present in the remote directory to connect. The cookie is checked both before
+// and after the connection. /tmp is sticky, and different Chrome sessions use
+// different cookies. Thus, a matching cookie before and after means the
+// connection was to a directory with a valid cookie.
+//
// We also have a lock file, which is a symlink to a non-existent destination.
// The destination is a string containing the hostname and process id of
// chrome's browser process, eg. "SingletonLock -> example.com-9156". When the
@@ -55,6 +65,7 @@
#include "base/path_service.h"
#include "base/platform_thread.h"
#include "base/process_util.h"
+#include "base/rand_util.h"
#include "base/safe_strerror_posix.h"
#include "base/stl_util-inl.h"
#include "base/string_number_conversions.h"
@@ -190,43 +201,56 @@ ssize_t ReadFromSocket(int fd, char *buf, size_t bufsize, int timeout) {
return bytes_read;
}
-// Set up a socket and sockaddr appropriate for messaging.
-void SetupSocket(const std::string& path, int* sock, struct sockaddr_un* addr) {
- *sock = socket(PF_UNIX, SOCK_STREAM, 0);
- PCHECK(*sock >= 0) << "socket() failed";
-
- int rv = SetNonBlocking(*sock);
- DCHECK_EQ(0, rv) << "Failed to make non-blocking socket.";
- rv = SetCloseOnExec(*sock);
- DCHECK_EQ(0, rv) << "Failed to set CLOEXEC on socket.";
-
+// Set up a sockaddr appropriate for messaging.
+void SetupSockAddr(const std::string& path, struct sockaddr_un* addr) {
addr->sun_family = AF_UNIX;
CHECK(path.length() < arraysize(addr->sun_path))
<< "Socket path too long: " << path;
base::strlcpy(addr->sun_path, path.c_str(), arraysize(addr->sun_path));
}
-// Read a symbol link, return empty string if given path is not a symbol link.
-std::string ReadLink(const std::string& path) {
- struct stat statbuf;
+// Set up a socket appropriate for messaging.
+int SetupSocketOnly() {
+ int sock = socket(PF_UNIX, SOCK_STREAM, 0);
+ PCHECK(sock >= 0) << "socket() failed";
- if (lstat(path.c_str(), &statbuf) < 0) {
- DCHECK_EQ(errno, ENOENT);
- return std::string();
+ int rv = SetNonBlocking(sock);
+ DCHECK_EQ(0, rv) << "Failed to make non-blocking socket.";
+ rv = SetCloseOnExec(sock);
+ DCHECK_EQ(0, rv) << "Failed to set CLOEXEC on socket.";
+
+ return sock;
+}
+
+// Set up a socket and sockaddr appropriate for messaging.
+void SetupSocket(const std::string& path, int* sock, struct sockaddr_un* addr) {
+ *sock = SetupSocketOnly();
+ SetupSockAddr(path, addr);
+}
+
+// Read a symbolic link, return empty string if given path is not a
+// symbol link. This version does not interpret the errno, leaving
+// the caller to do so.
+bool ReadLinkSilent(const std::string& path, std::string* output) {
+ char buf[PATH_MAX];
+ ssize_t len = readlink(path.c_str(), buf, PATH_MAX);
+ if (len >= 0) {
+ output->assign(buf, len);
+ return true;
}
+ output->clear();
+ return false;
+}
- if (S_ISLNK(statbuf.st_mode)) {
- char buf[PATH_MAX + 1];
- ssize_t len = readlink(path.c_str(), buf, PATH_MAX);
- if (len > 0) {
- buf[len] = '\0';
- return std::string(buf);
- } else {
+// Read a symbolic link, return empty string if given path is not a symbol link.
+std::string ReadLink(const std::string& path) {
+ std::string target;
+ if (!ReadLinkSilent(path, &target)) {
+ // The only errno that should occur is ENOENT.
+ if (errno != 0 && errno != ENOENT)
PLOG(ERROR) << "readlink(" << path << ") failed";
- }
}
-
- return std::string();
+ return target;
}
// Unlink a path. Return true on success.
@@ -238,6 +262,23 @@ bool UnlinkPath(const std::string& path) {
return rv == 0;
}
+// Create a symlink. Returns true on success.
+bool SymlinkPath(const std::string& target, const std::string& path) {
+ if (symlink(target.c_str(), path.c_str()) < 0) {
+ // Double check the value in case symlink suceeded but we got an incorrect
+ // failure due to NFS packet loss & retry.
+ int saved_errno = errno;
+ if (ReadLink(path) != target) {
+ // If we failed to create the lock, most likely another instance won the
+ // startup race.
+ errno = saved_errno;
+ PLOG(ERROR) << "Failed to create " << path;
+ return false;
+ }
+ }
+ return true;
+}
+
// Extract the hostname and pid from the lock symlink.
// Returns true if the lock existed.
bool ParseLockPath(const std::string& path,
@@ -247,7 +288,7 @@ bool ParseLockPath(const std::string& path,
if (real_path.empty())
return false;
- std::string::size_type pos = real_path.rfind('-');
+ std::string::size_type pos = real_path.rfind(kLockDelimiter);
// If the path is not a symbolic link, or doesn't contain what we expect,
// bail.
@@ -336,15 +377,83 @@ bool KillProcessByLockPath(const std::string& path) {
return true;
}
-// A helper class to close a socket automatically.
-class SocketCloser {
+// A helper class to hold onto a socket.
+class ScopedSocket {
public:
- explicit SocketCloser(int fd) : fd_(fd) { }
- ~SocketCloser() { CloseSocket(fd_); }
+ ScopedSocket() : fd_(-1) { Reset(); }
+ ~ScopedSocket() { Close(); }
+ int fd() { return fd_; }
+ void Reset() {
+ Close();
+ fd_ = SetupSocketOnly();
+ }
+ void Close() {
+ if (fd_ >= 0)
+ CloseSocket(fd_);
+ fd_ = -1;
+ }
private:
int fd_;
};
+// Returns a random string for uniquifying profile connections.
+std::string GenerateCookie() {
+ return base::Uint64ToString(base::RandUint64());
+}
+
+bool CheckCookie(const FilePath& path, const std::string& cookie) {
+ return (cookie == ReadLink(path.value()));
+}
+
+bool ConnectSocket(ScopedSocket* socket,
+ const FilePath& socket_path,
+ const FilePath& cookie_path) {
+ std::string socket_target;
+ if (ReadLinkSilent(socket_path.value(), &socket_target)) {
+ // It's a symlink. Read the cookie.
+ std::string cookie = ReadLink(cookie_path.value());
+ if (cookie.empty())
+ return false;
+ FilePath remote_cookie = FilePath(socket_target).DirName().
+ Append(chrome::kSingletonCookieFilename);
+ // Verify the cookie before connecting.
+ if (!CheckCookie(remote_cookie, cookie))
+ return false;
+ // Now we know the directory was (at that point) created by the profile
+ // owner. Try to connect.
+ sockaddr_un addr;
+ SetupSockAddr(socket_path.value(), &addr);
+ int ret = HANDLE_EINTR(connect(socket->fd(),
+ reinterpret_cast<sockaddr*>(&addr),
+ sizeof(addr)));
+ if (ret != 0)
+ return false;
+ // Check the cookie again. We only link in /tmp, which is sticky, so, if the
+ // directory is still correct, it must have been correct in-between when we
+ // connected. POSIX, sadly, lacks a connectat().
+ if (!CheckCookie(remote_cookie, cookie)) {
+ socket->Reset();
+ return false;
+ }
+ // Success!
+ return true;
+ } else if (errno == EINVAL) {
+ // It exists, but is not a symlink (or some other error we detect
+ // later). Just connect to it directly; this is an older version of Chrome.
+ sockaddr_un addr;
+ SetupSockAddr(socket_path.value(), &addr);
+ int ret = HANDLE_EINTR(connect(socket->fd(),
+ reinterpret_cast<sockaddr*>(&addr),
+ sizeof(addr)));
+ return (ret == 0);
+ } else {
+ // File is missing, or other error.
+ if (errno != ENOENT)
+ PLOG(ERROR) << "readlink failed";
+ return false;
+ }
+}
+
} // namespace
///////////////////////////////////////////////////////////////////////////////
@@ -653,6 +762,7 @@ ProcessSingleton::ProcessSingleton(const FilePath& user_data_dir)
ALLOW_THIS_IN_INITIALIZER_LIST(watcher_(new LinuxWatcher(this))) {
socket_path_ = user_data_dir.Append(chrome::kSingletonSocketFilename);
lock_path_ = user_data_dir.Append(chrome::kSingletonLockFilename);
+ cookie_path_ = user_data_dir.Append(chrome::kSingletonCookieFilename);
}
ProcessSingleton::~ProcessSingleton() {
@@ -670,19 +780,10 @@ ProcessSingleton::NotifyResult ProcessSingleton::NotifyOtherProcessWithTimeout(
bool kill_unresponsive) {
DCHECK_GE(timeout_seconds, 0);
- int socket;
- sockaddr_un addr;
- SetupSocket(socket_path_.value(), &socket, &addr);
-
- // It'll close the socket automatically when exiting this method.
- SocketCloser socket_closer(socket);
-
+ ScopedSocket socket;
for (int retries = 0; retries <= timeout_seconds; ++retries) {
- // Connecting to the socket
- int ret = HANDLE_EINTR(connect(socket,
- reinterpret_cast<sockaddr*>(&addr),
- sizeof(addr)));
- if (ret == 0)
+ // Try to connect to the socket.
+ if (ConnectSocket(&socket, socket_path_, cookie_path_))
break;
// If we're in a race with another process, they may be in Create() and have
@@ -733,7 +834,7 @@ ProcessSingleton::NotifyResult ProcessSingleton::NotifyOtherProcessWithTimeout(
}
timeval timeout = {timeout_seconds, 0};
- setsockopt(socket, SOL_SOCKET, SO_SNDTIMEO, &timeout, sizeof(timeout));
+ setsockopt(socket.fd(), SOL_SOCKET, SO_SNDTIMEO, &timeout, sizeof(timeout));
// Found another process, prepare our command line
// format is "START\0<current dir>\0<argv[0]>\0...\0<argv[n]>".
@@ -753,21 +854,21 @@ ProcessSingleton::NotifyResult ProcessSingleton::NotifyOtherProcessWithTimeout(
}
// Send the message
- if (!WriteToSocket(socket, to_send.data(), to_send.length())) {
+ if (!WriteToSocket(socket.fd(), to_send.data(), to_send.length())) {
// Try to kill the other process, because it might have been dead.
if (!kill_unresponsive || !KillProcessByLockPath(lock_path_.value()))
return PROFILE_IN_USE;
return PROCESS_NONE;
}
- if (shutdown(socket, SHUT_WR) < 0)
+ if (shutdown(socket.fd(), SHUT_WR) < 0)
PLOG(ERROR) << "shutdown() failed";
// Read ACK message from the other process. It might be blocked for a certain
// timeout, to make sure the other process has enough time to return ACK.
char buf[kMaxACKMessageLength + 1];
ssize_t len =
- ReadFromSocket(socket, buf, kMaxACKMessageLength, timeout_seconds);
+ ReadFromSocket(socket.fd(), buf, kMaxACKMessageLength, timeout_seconds);
// Failed to read ACK, the other process might have been frozen.
if (len <= 0) {
@@ -834,25 +935,41 @@ bool ProcessSingleton::Create() {
// Create symbol link before binding the socket, to ensure only one instance
// can have the socket open.
- if (symlink(symlink_content.c_str(), lock_path_.value().c_str()) < 0) {
- // Double check the value in case symlink suceeded but we got an incorrect
- // failure due to NFS packet loss & retry.
- int saved_errno = errno;
- if (ReadLink(lock_path_.value()) != symlink_content) {
+ if (!SymlinkPath(symlink_content, lock_path_.value())) {
// If we failed to create the lock, most likely another instance won the
// startup race.
- errno = saved_errno;
- PLOG(ERROR) << "Failed to create " << lock_path_.value();
return false;
- }
}
- SetupSocket(socket_path_.value(), &sock, &addr);
-
+ // Create the socket file somewhere in /tmp which is usually mounted as a
+ // normal filesystem. Some network filesystems (notably AFS) are screwy and
+ // do not support Unix domain sockets.
+ if (!socket_dir_.CreateUniqueTempDir()) {
+ LOG(ERROR) << "Failed to create socket directory.";
+ return false;
+ }
+ // Setup the socket symlink and the two cookies.
+ FilePath socket_target_path =
+ socket_dir_.path().Append(chrome::kSingletonSocketFilename);
+ std::string cookie = GenerateCookie();
+ FilePath remote_cookie_path =
+ socket_dir_.path().Append(chrome::kSingletonCookieFilename);
UnlinkPath(socket_path_.value());
+ UnlinkPath(cookie_path_.value());
+ if (!SymlinkPath(socket_target_path.value(), socket_path_.value()) ||
+ !SymlinkPath(cookie, cookie_path_.value()) ||
+ !SymlinkPath(cookie, remote_cookie_path.value())) {
+ // We've already locked things, so we can't have lost the startup race,
+ // but something doesn't like us.
+ LOG(ERROR) << "Failed to create symlinks.";
+ socket_dir_.Delete();
+ return false;
+ }
+
+ SetupSocket(socket_target_path.value(), &sock, &addr);
if (bind(sock, reinterpret_cast<sockaddr*>(&addr), sizeof(addr)) < 0) {
- PLOG(ERROR) << "Failed to bind() " << socket_path_.value();
+ PLOG(ERROR) << "Failed to bind() " << socket_target_path.value();
CloseSocket(sock);
return false;
}
@@ -874,5 +991,7 @@ bool ProcessSingleton::Create() {
}
void ProcessSingleton::Cleanup() {
+ UnlinkPath(socket_path_.value());
+ UnlinkPath(cookie_path_.value());
UnlinkPath(lock_path_.value());
}
diff --git a/chrome/browser/process_singleton_linux_uitest.cc b/chrome/browser/process_singleton_linux_uitest.cc
index a770023..fc3fca0 100644
--- a/chrome/browser/process_singleton_linux_uitest.cc
+++ b/chrome/browser/process_singleton_linux_uitest.cc
@@ -35,6 +35,7 @@ class ProcessSingletonLinuxTest : public UITest {
UITest::SetUp();
lock_path_ = user_data_dir().Append(chrome::kSingletonLockFilename);
socket_path_ = user_data_dir().Append(chrome::kSingletonSocketFilename);
+ cookie_path_ = user_data_dir().Append(chrome::kSingletonCookieFilename);
}
virtual void TearDown() {
@@ -53,6 +54,7 @@ class ProcessSingletonLinuxTest : public UITest {
FilePath lock_path_;
FilePath socket_path_;
+ FilePath cookie_path_;
};
ProcessSingleton* CreateProcessSingleton() {
@@ -106,13 +108,29 @@ TEST_F(ProcessSingletonLinuxTest, CheckSocketFile) {
struct stat statbuf;
ASSERT_EQ(0, lstat(lock_path_.value().c_str(), &statbuf));
ASSERT_TRUE(S_ISLNK(statbuf.st_mode));
- char buf[PATH_MAX + 1];
+ char buf[PATH_MAX];
ssize_t len = readlink(lock_path_.value().c_str(), buf, PATH_MAX);
ASSERT_GT(len, 0);
- buf[len] = '\0';
ASSERT_EQ(0, lstat(socket_path_.value().c_str(), &statbuf));
+ ASSERT_TRUE(S_ISLNK(statbuf.st_mode));
+
+ len = readlink(socket_path_.value().c_str(), buf, PATH_MAX);
+ ASSERT_GT(len, 0);
+ FilePath socket_target_path = FilePath(std::string(buf, len));
+
+ ASSERT_EQ(0, lstat(socket_target_path.value().c_str(), &statbuf));
ASSERT_TRUE(S_ISSOCK(statbuf.st_mode));
+
+ len = readlink(cookie_path_.value().c_str(), buf, PATH_MAX);
+ ASSERT_GT(len, 0);
+ std::string cookie(buf, len);
+
+ FilePath remote_cookie_path = socket_target_path.DirName().
+ Append(chrome::kSingletonCookieFilename);
+ len = readlink(remote_cookie_path.value().c_str(), buf, PATH_MAX);
+ ASSERT_GT(len, 0);
+ EXPECT_EQ(cookie, std::string(buf, len));
}
#if defined(OS_LINUX) && defined(TOOLKIT_VIEWS)
@@ -247,3 +265,37 @@ TEST_F(ProcessSingletonLinuxTest, CreateFailsWithExistingBrowser) {
scoped_ptr<ProcessSingleton> process_singleton(CreateProcessSingleton());
EXPECT_FALSE(process_singleton->Create());
}
+
+// Test that Create fails when another browser is using the profile directory
+// but with the old socket location.
+TEST_F(ProcessSingletonLinuxTest, CreateChecksCompatibilitySocket) {
+ scoped_ptr<ProcessSingleton> process_singleton(CreateProcessSingleton());
+
+ // Do some surgery so as to look like the old configuration.
+ char buf[PATH_MAX];
+ ssize_t len = readlink(socket_path_.value().c_str(), buf, sizeof(buf));
+ ASSERT_GT(len, 0);
+ FilePath socket_target_path = FilePath(std::string(buf, len));
+ ASSERT_EQ(0, unlink(socket_path_.value().c_str()));
+ ASSERT_EQ(0, rename(socket_target_path.value().c_str(),
+ socket_path_.value().c_str()));
+ ASSERT_EQ(0, unlink(cookie_path_.value().c_str()));
+
+ EXPECT_FALSE(process_singleton->Create());
+}
+
+// Test that we fail when lock says process is on another host and we can't
+// notify it over the socket before of a bad cookie.
+TEST_F(ProcessSingletonLinuxTest, NotifyOtherProcessOrCreate_BadCookie) {
+ // Change the cookie.
+ EXPECT_EQ(0, unlink(cookie_path_.value().c_str()));
+ EXPECT_EQ(0, symlink("INCORRECTCOOKIE", cookie_path_.value().c_str()));
+
+ // Also change the hostname, so the remote does not retry.
+ EXPECT_EQ(0, unlink(lock_path_.value().c_str()));
+ EXPECT_EQ(0, symlink("FAKEFOOHOST-1234", lock_path_.value().c_str()));
+
+ std::string url("about:blank");
+ EXPECT_EQ(ProcessSingleton::PROFILE_IN_USE,
+ NotifyOtherProcessOrCreate(url, action_timeout_ms()));
+}
diff --git a/chrome/common/chrome_constants.cc b/chrome/common/chrome_constants.cc
index 2cedf3d..2047798 100644
--- a/chrome/common/chrome_constants.cc
+++ b/chrome/common/chrome_constants.cc
@@ -90,8 +90,7 @@ const FilePath::CharType kHistoryFilename[] = FPL("History");
const FilePath::CharType kLocalStateFilename[] = FPL("Local State");
const FilePath::CharType kPreferencesFilename[] = FPL("Preferences");
const FilePath::CharType kSafeBrowsingFilename[] = FPL("Safe Browsing Bloom");
-// WARNING: SingletonSocket can't contain spaces, because otherwise
-// chrome_process_util_linux would be broken.
+const FilePath::CharType kSingletonCookieFilename[] = FPL("SingletonCookie");
const FilePath::CharType kSingletonSocketFilename[] = FPL("SingletonSocket");
const FilePath::CharType kSingletonLockFilename[] = FPL("SingletonLock");
const FilePath::CharType kThumbnailsFilename[] = FPL("Thumbnails");
diff --git a/chrome/common/chrome_constants.h b/chrome/common/chrome_constants.h
index 9c70974..6510409b 100644
--- a/chrome/common/chrome_constants.h
+++ b/chrome/common/chrome_constants.h
@@ -51,6 +51,7 @@ extern const FilePath::CharType kHistoryFilename[];
extern const FilePath::CharType kLocalStateFilename[];
extern const FilePath::CharType kPreferencesFilename[];
extern const FilePath::CharType kSafeBrowsingFilename[];
+extern const FilePath::CharType kSingletonCookieFilename[];
extern const FilePath::CharType kSingletonSocketFilename[];
extern const FilePath::CharType kSingletonLockFilename[];
extern const FilePath::CharType kThumbnailsFilename[];