Files
libreoffice/shell/source/unix/sysshell/recently_used_file.cxx

184 lines
4.7 KiB
C++
Raw Normal View History

/*************************************************************************
*
* OpenOffice.org - a multi-platform office productivity suite
*
* $RCSfile: recently_used_file.cxx,v $
*
* $Revision: 1.6 $
*
* last change: $Author: obo $ $Date: 2006-09-17 01:42:27 $
*
* The Contents of this file are made available subject to
* the terms of GNU Lesser General Public License Version 2.1.
*
*
* GNU Lesser General Public License Version 2.1
* =============================================
* Copyright 2005 by Sun Microsystems, Inc.
* 901 San Antonio Road, Palo Alto, CA 94303, USA
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License version 2.1, as published by the Free Software Foundation.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston,
* MA 02111-1307 USA
*
************************************************************************/
// MARKER(update_precomp.py): autogen include statement, do not remove
#include "precompiled_shell.hxx"
#ifndef INCLUDED_RECENTLY_USED_FILE
#include "recently_used_file.hxx"
#endif
#ifndef _RTL_USTRING_HXX_
#include <rtl/ustring.hxx>
#endif
#ifndef _OSL_PROCESS_H_
#include <osl/process.h>
#endif
#ifndef _OSL_SECURITY_HXX_
#include <osl/security.hxx>
#endif
#ifndef _OSL_THREAD_H_
#include <osl/thread.h>
#endif
#ifndef _OSL_FILE_HXX_
#include <osl/file.hxx>
#endif
#include <sys/file.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
const rtl::OUString RECENTLY_USED_FILE_NAME = rtl::OUString::createFromAscii(".recently-used");
const rtl::OUString SLASH = rtl::OUString::createFromAscii("/");
namespace /* private */ {
inline void ensure_final_slash(/*inout*/ rtl::OUString& path)
{
if ((path.getLength() > 0) &&
(SLASH.pData->buffer[0] != path.pData->buffer[path.getLength() - 1]))
path += SLASH;
}
} // namespace private
//------------------------------------------------
recently_used_file::recently_used_file() :
file_(NULL)
{
osl::Security sec;
rtl::OUString homedir_url;
if (sec.getHomeDir(homedir_url))
{
rtl::OUString homedir;
osl::FileBase::getSystemPathFromFileURL(homedir_url, homedir);
rtl::OUString rufn = homedir;
ensure_final_slash(rufn);
rufn += RECENTLY_USED_FILE_NAME;
rtl::OString tmp =
rtl::OUStringToOString(rufn, osl_getThreadTextEncoding());
file_ = fopen(tmp.getStr(), "r+");
/* create if not exist */
if (NULL == file_) {
mode_t umask_ = umask(S_IRGRP|S_IWGRP|S_IROTH|S_IWOTH);
file_ = fopen(tmp.getStr(), "w+");
umask(umask_);
}
if (NULL == file_)
throw "I/O error opening ~/.recently-used";
#if defined(MACOSX) && (BUILD_OS_MINOR == 2)
if (flock(fileno(file_),(LOCK_EX | LOCK_NB)) != 0)
#else
if (lockf(fileno(file_), F_LOCK, 0) != 0)
#endif
{
fclose(file_);
throw "Cannot lock ~/.recently-used";
}
}
else
throw "Cannot determine user home directory";
}
//------------------------------------------------
recently_used_file::~recently_used_file()
{
#if defined(MACOSX) && (BUILD_OS_MINOR == 2)
flock(fileno(file_),LOCK_UN);
#else
lockf(fileno(file_), F_ULOCK, 0);
#endif
fclose(file_);
}
//------------------------------------------------
void recently_used_file::reset() const
{
rewind(file_);
}
//------------------------------------------------
void recently_used_file::flush()
{
fflush(file_);
}
//------------------------------------------------
void recently_used_file::truncate(off_t length)
{
ftruncate(fileno(file_), length);
}
//------------------------------------------------
size_t recently_used_file::read(char* buffer, size_t size) const
{
size_t r = fread(reinterpret_cast<void*>(buffer), sizeof(char), size, file_);
if ((r < size) && ferror(file_))
throw "I/O error: read failed";
return r;
}
//----------------------------
void recently_used_file::write(const char* buffer, size_t size) const
{
if (size != fwrite(reinterpret_cast<const void*>(buffer), sizeof(char), size, file_))
throw "I/O error: write failed";
}
//----------------------------
bool recently_used_file::eof() const
{
return feof(file_);
}