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
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
|
// Copyright (c) 2009 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.
// On Linux, when the user tries to launch a second copy of chrome, we check
// for a socket in the user's profile directory. If the socket file is open we
// send a message to the first chrome browser process with the current
// directory and second process command line flags. The second process then
// exits.
#include "chrome/browser/process_singleton.h"
#include <errno.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <sys/un.h>
#include "base/base_paths.h"
#include "base/command_line.h"
#include "base/eintr_wrapper.h"
#include "base/logging.h"
#include "base/message_loop.h"
#include "base/path_service.h"
#include "base/string_util.h"
#include "chrome/browser/browser_init.h"
#include "chrome/browser/browser_process.h"
#include "chrome/browser/chrome_thread.h"
#include "chrome/browser/profile.h"
#include "chrome/browser/profile_manager.h"
#include "chrome/common/chrome_constants.h"
#include "chrome/common/chrome_paths.h"
namespace {
const char* kStartToken = "START";
const char kTokenDelimiter = '\0';
}
///////////////////////////////////////////////////////////////////////////////
// ProcessSingleton::LinuxWatcher
// A helper class for a Linux specific implementation of the process singleton.
// This class sets up a listener on the singleton socket and handles parsing
// messages that come in on the singleton socket.
class ProcessSingleton::LinuxWatcher
: public MessageLoopForIO::Watcher,
public base::RefCountedThreadSafe<ProcessSingleton::LinuxWatcher> {
public:
class SocketReader : public MessageLoopForIO::Watcher {
public:
SocketReader(ProcessSingleton::LinuxWatcher* parent,
MessageLoop* ui_message_loop)
: parent_(parent),
ui_message_loop_(ui_message_loop) {}
virtual ~SocketReader() {}
MessageLoopForIO::FileDescriptorWatcher* fd_reader() {
return &fd_reader_;
}
// MessageLoopForIO::Watcher impl.
virtual void OnFileCanReadWithoutBlocking(int fd);
virtual void OnFileCanWriteWithoutBlocking(int fd) {
// ProcessSingleton only watches for accept (read) events.
NOTREACHED();
}
private:
MessageLoopForIO::FileDescriptorWatcher fd_reader_;
// The ProcessSingleton::LinuxWatcher that owns us.
ProcessSingleton::LinuxWatcher* parent_;
// A reference to the UI message loop.
MessageLoop* ui_message_loop_;
DISALLOW_COPY_AND_ASSIGN(SocketReader);
};
// We expect to only be constructed on the UI thread.
explicit LinuxWatcher(ProcessSingleton* parent)
: ui_message_loop_(MessageLoop::current()),
parent_(parent) {
}
virtual ~LinuxWatcher() {}
// This method determines if we should use the same process and if we should,
// opens a new browser tab. This runs on the UI thread.
void HandleMessage(std::string current_dir, std::vector<std::string> argv);
MessageLoopForIO::FileDescriptorWatcher* fd_watcher() {
return &fd_watcher_;
}
// MessageLoopForIO::Watcher impl. These run on the IO thread.
virtual void OnFileCanReadWithoutBlocking(int fd);
virtual void OnFileCanWriteWithoutBlocking(int fd) {
// ProcessSingleton only watches for accept (read) events.
NOTREACHED();
}
private:
MessageLoopForIO::FileDescriptorWatcher fd_watcher_;
// A reference to the UI message loop (i.e., the message loop we were
// constructed on).
MessageLoop* ui_message_loop_;
// The ProcessSingleton that owns us.
ProcessSingleton* parent_;
// The MessageLoopForIO::Watcher that actually reads data from the socket.
// TODO(tc): We shouldn't need to keep a pointer to this. That way we can
// handle multiple concurrent requests.
scoped_ptr<SocketReader> reader_;
DISALLOW_COPY_AND_ASSIGN(LinuxWatcher);
};
void ProcessSingleton::LinuxWatcher::OnFileCanReadWithoutBlocking(int fd) {
// Accepting incoming client.
sockaddr_un from;
socklen_t from_len = sizeof(from);
int connection_socket = HANDLE_EINTR(accept(
fd, reinterpret_cast<sockaddr*>(&from), &from_len));
if (-1 == connection_socket) {
LOG(ERROR) << "accept() failed: " << strerror(errno);
return;
}
reader_.reset(new SocketReader(this, ui_message_loop_));
// Wait for reads.
MessageLoopForIO::current()->WatchFileDescriptor(connection_socket,
true, MessageLoopForIO::WATCH_READ, reader_->fd_reader(), reader_.get());
}
void ProcessSingleton::LinuxWatcher::HandleMessage(std::string current_dir,
std::vector<std::string> argv) {
// Ignore the request if the browser process is already in shutdown path.
if (!g_browser_process || g_browser_process->IsShuttingDown()) {
LOG(WARNING) << "Not handling interprocess notification as browser"
" is shutting down";
return;
}
// If locked, it means we are not ready to process this message because
// we are probably in a first run critical phase.
if (parent_->locked()) {
DLOG(WARNING) << "Browser is locked";
return;
}
CommandLine parsed_command_line(argv);
PrefService* prefs = g_browser_process->local_state();
DCHECK(prefs);
FilePath user_data_dir;
PathService::Get(chrome::DIR_USER_DATA, &user_data_dir);
ProfileManager* profile_manager = g_browser_process->profile_manager();
Profile* profile = profile_manager->GetDefaultProfile(user_data_dir);
if (!profile) {
// We should only be able to get here if the profile already exists and
// has been created.
NOTREACHED();
return;
}
// TODO(tc): Send an ACK response on success.
// Run the browser startup sequence again, with the command line of the
// signalling process.
FilePath current_dir_file_path(current_dir);
BrowserInit::ProcessCommandLine(parsed_command_line,
current_dir_file_path.ToWStringHack(),
false, profile, NULL);
}
///////////////////////////////////////////////////////////////////////////////
// ProcessSingleton::LinuxWatcher::SocketReader
//
void ProcessSingleton::LinuxWatcher::SocketReader::OnFileCanReadWithoutBlocking(
int fd) {
const int kMaxMessageLength = 32 * 1024;
char buf[kMaxMessageLength];
ssize_t rv = HANDLE_EINTR(read(fd, buf, sizeof(buf)));
if (rv < 0) {
LOG(ERROR) << "recv() failed: " << strerror(errno);
return;
}
// Validate the message. The shortest message is kStartToken\0x\0x\0
const ssize_t kMinMessageLength = strlen(kStartToken) + 5;
if (rv < kMinMessageLength) {
LOG(ERROR) << "Invalid socket message (wrong length):" << buf;
return;
}
std::string str(buf, rv);
std::vector<std::string> tokens;
SplitString(str, kTokenDelimiter, &tokens);
if (tokens.size() < 3 || tokens[0] != kStartToken) {
LOG(ERROR) << "Wrong message format: " << str;
return;
}
std::string current_dir = tokens[1];
// Remove the first two tokens. The remaining tokens should be the command
// line argv array.
tokens.erase(tokens.begin());
tokens.erase(tokens.begin());
// Return to the UI thread to handle opening a new browser tab.
ui_message_loop_->PostTask(FROM_HERE, NewRunnableMethod(
parent_,
&ProcessSingleton::LinuxWatcher::HandleMessage,
current_dir,
tokens));
fd_reader_.StopWatchingFileDescriptor();
}
///////////////////////////////////////////////////////////////////////////////
// ProcessSingleton
//
ProcessSingleton::ProcessSingleton(const FilePath& user_data_dir)
: locked_(false),
foreground_window_(NULL),
ALLOW_THIS_IN_INITIALIZER_LIST(watcher_(new LinuxWatcher(this))) {
socket_path_ = user_data_dir.Append(chrome::kSingletonSocketFilename);
}
ProcessSingleton::~ProcessSingleton() {
}
bool ProcessSingleton::NotifyOtherProcess() {
int socket;
sockaddr_un addr;
SetupSocket(&socket, &addr);
// Connecting to the socket
int ret = HANDLE_EINTR(connect(socket,
reinterpret_cast<sockaddr*>(&addr),
sizeof(addr)));
if (ret < 0)
return false; // Tell the caller there's nobody to notify.
timeval timeout = {20, 0};
setsockopt(socket, 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]>\0".
std::string to_send(kStartToken);
to_send.push_back(kTokenDelimiter);
FilePath current_dir;
if (!PathService::Get(base::DIR_CURRENT, ¤t_dir))
return false;
to_send.append(current_dir.value());
to_send.push_back(kTokenDelimiter);
const std::vector<std::string>& argv =
CommandLine::ForCurrentProcess()->argv();
for (std::vector<std::string>::const_iterator it = argv.begin();
it != argv.end(); ++it) {
to_send.append(*it);
to_send.push_back(kTokenDelimiter);
}
// Send the message
int rv = HANDLE_EINTR(write(socket, to_send.data(), to_send.length()));
if (rv < 0) {
if (errno == EAGAIN) {
// TODO(tc): We should kill the hung process here.
NOTIMPLEMENTED() << "browser process hung, don't know how to kill it";
return false;
}
LOG(ERROR) << "send() failed: " << strerror(errno);
return false;
}
// TODO(tc): We should wait for an ACK, and if we don't get it in a certain
// time period, kill the other process.
HANDLE_EINTR(close(socket));
// Assume the other process is handling the request.
return true;
}
void ProcessSingleton::Create() {
int sock;
sockaddr_un addr;
SetupSocket(&sock, &addr);
if (unlink(socket_path_.value().c_str()) < 0)
DCHECK_EQ(errno, ENOENT);
if (bind(sock, reinterpret_cast<sockaddr*>(&addr), sizeof(addr)) < 0)
LOG(ERROR) << "bind() failed: " << strerror(errno);
if (listen(sock, 5) < 0)
NOTREACHED() << "listen failed: " << strerror(errno);
// Normally we would use ChromeThread, but the IO thread hasn't started yet.
// Using g_browser_process, we start the thread so we can listen on the
// socket.
MessageLoop* ml = g_browser_process->io_thread()->message_loop();
DCHECK(ml);
// Watch for client connections on this socket.
static_cast<MessageLoopForIO*>(ml)->WatchFileDescriptor(sock, true,
MessageLoopForIO::WATCH_READ, watcher_->fd_watcher(), watcher_.get());
}
void ProcessSingleton::SetupSocket(int* sock, struct sockaddr_un* addr) {
*sock = socket(PF_UNIX, SOCK_STREAM, 0);
if (*sock < 0)
LOG(FATAL) << "socket() failed: " << strerror(errno);
addr->sun_family = AF_UNIX;
if (socket_path_.value().length() > sizeof(addr->sun_path) - 1)
LOG(FATAL) << "Socket path too long: " << socket_path_.value();
base::strlcpy(addr->sun_path, socket_path_.value().c_str(),
sizeof(addr->sun_path));
}
|