8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-27 05:23:02 +01:00
firebird-mirror/src/common/classes/TempFile.cpp

353 lines
6.9 KiB
C++

/*
* The contents of this file are subject to the Initial
* Developer's Public License Version 1.0 (the "License");
* you may not use this file except in compliance with the
* License. You may obtain a copy of the License at
* http://www.ibphoenix.com/main.nfs?a=ibphoenix&page=ibp_idpl.
*
* Software distributed under the License is distributed AS IS,
* WITHOUT WARRANTY OF ANY KIND, either express or implied.
* See the License for the specific language governing rights
* and limitations under the License.
*
* The Original Code was created by Dmitry Yemanov
* for the Firebird Open Source RDBMS project.
*
* Copyright (c) 2006 Dmitry Yemanov <dimitr@users.sf.net>
* and all contributors signed below.
*
* All Rights Reserved.
* Contributor(s): ______________________________________.
*/
#include "firebird.h"
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>
#if defined(WIN_NT)
#include <fcntl.h>
#include <io.h>
#include <process.h>
#include <share.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <sys/timeb.h>
#include <windows.h>
#ifndef INVALID_SET_FILE_POINTER
#define INVALID_SET_FILE_POINTER ((DWORD)-1)
#endif
#elif defined(HAVE_UNISTD_H)
#include <unistd.h>
#endif
#include "../jrd/gdsassert.h"
#include "../jrd/os/path_utils.h"
#include "../common/classes/init.h"
#include "../common/classes/TempFile.h"
namespace Firebird {
// Const definitions
static const char* ENV_VAR = "FIREBIRD_TMP";
static const char* DEFAULT_PATH =
#if defined(UNIX)
"/tmp/";
#elif defined(WIN_NT)
"c:\\temp\\";
#else
NULL;
#endif
static const char* const NAME_PATTERN = "XXXXXX";
static const char* const NAME_LETTERS = "abcdefghijklmnopqrstuvwxyz0123456789";
static const size_t MAX_TRIES = 256;
// we need a class here only to return memory on shutdown and avoid
// false memory leak reports
static Firebird::InitInstance<ZeroBuffer> zeros;
//
// TempFile::getTempPath
//
// Returns a pathname to the system temporary directory
//
PathName TempFile::getTempPath()
{
const char* env_temp = getenv(ENV_VAR);
PathName path = env_temp ? env_temp : "";
if (path.empty())
{
#if defined(WIN_NT)
char temp_dir[MAXPATHLEN];
// this checks "TEMP" and "TMP" environment variables
const int len = GetTempPath(sizeof(temp_dir), temp_dir);
if (len && len < sizeof(temp_dir))
{
path = temp_dir;
}
#else
env_temp = getenv("TMP");
path = env_temp ? env_temp : "";
#endif
}
if (path.empty())
{
path = DEFAULT_PATH;
}
fb_assert(path.length());
return path;
}
//
// TempFile::create
//
// Creates a temporary file and returns its name
//
PathName TempFile::create(const PathName& prefix, const PathName& directory)
{
PathName filename;
try {
TempFile file(*getDefaultMemoryPool(), prefix, directory, false);
filename = file.getName();
}
catch (const Exception&)
{} // do nothing
return filename;
}
//
// TempFile::init
//
// Creates temporary file with a unique filename
//
void TempFile::init(const PathName& directory, const PathName& prefix)
{
// set up temporary directory, if not specified
filename = directory;
if (filename.empty())
{
filename = getTempPath();
}
PathUtils::ensureSeparator(filename);
#if defined(WIN_NT)
_timeb t;
_ftime(&t);
__int64 randomness = t.time;
randomness *= 1000;
randomness += t.millitm;
PathName suffix = NAME_PATTERN;
for (int tries = 0; tries < MAX_TRIES; tries++)
{
PathName name = filename + prefix;
__int64 temp = randomness;
for (size_t i = 0; i < suffix.length(); i++)
{
suffix[i] = NAME_LETTERS[temp % (strlen(NAME_LETTERS))];
temp /= strlen(NAME_LETTERS);
}
name += suffix;
DWORD attributes = FILE_ATTRIBUTE_NORMAL | FILE_ATTRIBUTE_TEMPORARY | FILE_FLAG_RANDOM_ACCESS;
if (doUnlink)
{
attributes |= FILE_FLAG_DELETE_ON_CLOSE;
}
handle = CreateFile(name.c_str(), GENERIC_READ | GENERIC_WRITE,
0, NULL, CREATE_NEW, attributes, NULL);
if (handle != INVALID_HANDLE_VALUE)
{
filename = name;
break;
}
randomness++;
}
if (handle == INVALID_HANDLE_VALUE)
{
system_error::raise("CreateFile");
}
#else
filename += prefix;
filename += NAME_PATTERN;
#ifdef HAVE_MKSTEMP
handle = (IPTR) mkstemp(filename.begin());
#else
if (!mktemp(filename.begin()))
{
system_error::raise("mktemp");
}
do {
handle = open(filename.c_str(), O_RDWR | O_EXCL | O_CREAT);
} while (handle == -1 && errno == EINTR);
#endif
if (handle == -1)
{
system_error::raise("open");
}
if (doUnlink)
{
::unlink(filename.c_str());
}
#endif
doUnlink = false;
}
//
// TempFile::~TempFile
//
// Destructor
//
TempFile::~TempFile()
{
#if defined(WIN_NT)
CloseHandle(handle);
#else
::close(handle);
#endif
if (doUnlink)
{
::unlink(filename.c_str());
}
}
//
// TempFile::seek
//
// Performs a positioning operation
//
void TempFile::seek(const offset_t offset)
{
if (position == offset)
return;
#if defined(WIN_NT)
LARGE_INTEGER liOffset;
liOffset.QuadPart = offset;
const DWORD seek_result =
SetFilePointer(handle, (LONG) liOffset.LowPart, &liOffset.HighPart, FILE_BEGIN);
if (seek_result == INVALID_SET_FILE_POINTER && GetLastError() != NO_ERROR)
{
system_error::raise("SetFilePointer");
}
#else
const off_t seek_result = ::lseek(handle, (off_t) offset, SEEK_SET);
if (seek_result == (off_t) -1)
{
system_error::raise("lseek");
}
#endif
position = offset;
if (position > size)
size = position;
}
//
// TempFile::extend
//
// Increases the file size
//
void TempFile::extend(size_t delta)
{
const char* const buffer = zeros().getBuffer();
const size_t bufferSize = zeros().getSize();
const offset_t newSize = size + delta;
for (offset_t offset = size; offset < newSize; offset += bufferSize)
{
const size_t length = MIN(newSize - offset, bufferSize);
write(offset, buffer, length);
}
}
//
// TempFile::read
//
// Reads bytes from file
//
size_t TempFile::read(offset_t offset, void* buffer, size_t length)
{
fb_assert(offset + length <= size);
seek(offset);
#if defined(WIN_NT)
DWORD bytes = 0;
if (!ReadFile(handle, buffer, length, &bytes, NULL) || bytes != length)
{
system_error::raise("ReadFile");
}
#else
const int bytes = ::read(handle, buffer, length);
if (bytes < 0 || size_t(bytes) != length)
{
system_error::raise("read");
}
#endif
position += bytes;
return bytes;
}
//
// TempFile::write
//
// Writes bytes to file
//
size_t TempFile::write(offset_t offset, const void* buffer, size_t length)
{
fb_assert(offset <= size);
seek(offset);
#if defined(WIN_NT)
DWORD bytes = 0;
if (!WriteFile(handle, buffer, length, &bytes, NULL) || bytes != length)
{
system_error::raise("WriteFile");
}
#else
const int bytes = ::write(handle, buffer, length);
if (bytes < 0 || size_t(bytes) != length)
{
system_error::raise("write");
}
#endif
position += bytes;
if (position > size)
size = position;
return bytes;
}
//
// TempFile::unlink
//
// Unlinks the file
//
void TempFile::unlink()
{
#if defined(WIN_NT)
doUnlink = true;
#else
::unlink(filename.c_str());
#endif
}
} // namespace Firebird