Home | History | Annotate | Download | only in importer
      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 (at) netscape.com>
     40 *   Brendan Eich <brendan (at) mozilla.org>
     41 *   Colin Blake <colin (at) theblakes.com>
     42 *   Javier Pedemonte <pedemont (at) us.ibm.com>
     43 *   Mats Palmgren <mats.palmgren (at) 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 
     59 void FirefoxProfileLock::Init() {
     60   lock_fd_ = -1;
     61 }
     62 
     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 
     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 
     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).
     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