• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 2009 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #include "chrome/browser/importer/firefox_profile_lock.h"
6 
7 #include <errno.h>
8 #include <fcntl.h>
9 #include <sys/stat.h>
10 #include <sys/types.h>
11 
12 #include "base/file_util.h"
13 
14 // This class is based on Firefox code in:
15 //   profile/dirserviceprovider/src/nsProfileLock.cpp
16 // The license block is:
17 
18 /* ***** BEGIN LICENSE BLOCK *****
19 * Version: MPL 1.1/GPL 2.0/LGPL 2.1
20 *
21 * The contents of this file are subject to the Mozilla Public License Version
22 * 1.1 (the "License"); you may not use this file except in compliance with
23 * the License. You may obtain a copy of the License at
24 * http://www.mozilla.org/MPL/
25 *
26 * Software distributed under the License is distributed on an "AS IS" basis,
27 * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
28 * for the specific language governing rights and limitations under the
29 * License.
30 *
31 * The Original Code is mozilla.org code.
32 *
33 * The Initial Developer of the Original Code is
34 * Netscape Communications Corporation.
35 * Portions created by the Initial Developer are Copyright (C) 2002
36 * the Initial Developer. All Rights Reserved.
37 *
38 * Contributor(s):
39 *   Conrad Carlen <ccarlen@netscape.com>
40 *   Brendan Eich <brendan@mozilla.org>
41 *   Colin Blake <colin@theblakes.com>
42 *   Javier Pedemonte <pedemont@us.ibm.com>
43 *   Mats Palmgren <mats.palmgren@bredband.net>
44 *
45 * Alternatively, the contents of this file may be used under the terms of
46 * either the GNU General Public License Version 2 or later (the "GPL"), or
47 * the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
48 * in which case the provisions of the GPL or the LGPL are applicable instead
49 * of those above. If you wish to allow use of your version of this file only
50 * under the terms of either the GPL or the LGPL, and not to allow others to
51 * use your version of this file under the terms of the MPL, indicate your
52 * decision by deleting the provisions above and replace them with the notice
53 * and other provisions required by the GPL or the LGPL. If you do not delete
54 * the provisions above, a recipient may use your version of this file under
55 * the terms of any one of the MPL, the GPL or the LGPL.
56 *
57 * ***** END LICENSE BLOCK ***** */
58 
Init()59 void FirefoxProfileLock::Init() {
60   lock_fd_ = -1;
61 }
62 
Lock()63 void FirefoxProfileLock::Lock() {
64   if (HasAcquired())
65     return;
66 
67   bool fcntl_lock = LockWithFcntl();
68   if (!fcntl_lock) {
69     return;
70   } else if (!HasAcquired()) {
71     old_lock_file_ = lock_file_.DirName().Append(kOldLockFileName);
72     lock_fd_ = open(old_lock_file_.value().c_str(), O_CREAT | O_EXCL, 0644);
73   }
74 }
75 
Unlock()76 void FirefoxProfileLock::Unlock() {
77   if (!HasAcquired())
78     return;
79   close(lock_fd_);
80   lock_fd_ = -1;
81   file_util::Delete(old_lock_file_, false);
82 }
83 
HasAcquired()84 bool FirefoxProfileLock::HasAcquired() {
85   return (lock_fd_ >= 0);
86 }
87 
88 // This function tries to lock Firefox profile using fcntl(). The return
89 // value of this function together with HasAcquired() tells the current status
90 // of lock.
91 // if return == false: Another process has lock to the profile.
92 // if return == true && HasAcquired() == true: successfully acquired the lock.
93 // if return == false && HasAcquired() == false: Failed to acquire lock due
94 // to some error (so that we can try alternate method of profile lock).
LockWithFcntl()95 bool FirefoxProfileLock::LockWithFcntl() {
96   lock_fd_ = open(lock_file_.value().c_str(), O_WRONLY | O_CREAT | O_TRUNC,
97                   0666);
98   if (lock_fd_ == -1)
99     return true;
100 
101   struct flock lock;
102   lock.l_start = 0;
103   lock.l_len = 0;
104   lock.l_type = F_WRLCK;
105   lock.l_whence = SEEK_SET;
106   lock.l_pid = 0;
107 
108   struct flock testlock = lock;
109   if (fcntl(lock_fd_, F_GETLK, &testlock) == -1) {
110     close(lock_fd_);
111     lock_fd_ = -1;
112     return true;
113   } else if (fcntl(lock_fd_, F_SETLK, &lock) == -1) {
114     close(lock_fd_);
115     lock_fd_ = -1;
116     if (errno == EAGAIN || errno == EACCES)
117       return false;
118     else
119       return true;
120   } else {
121     // We have the lock.
122     return true;
123   }
124 }
125