2006-05-31 10:53:00 +02:00
|
|
|
/*
|
|
|
|
* 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"
|
|
|
|
|
2009-08-21 10:36:37 +02:00
|
|
|
#include "iberror.h"
|
2006-05-31 10:53:00 +02:00
|
|
|
#include "../common/config/config.h"
|
|
|
|
#include "../common/config/dir_list.h"
|
|
|
|
#include "../jrd/gdsassert.h"
|
|
|
|
#include "../jrd/gds_proto.h"
|
|
|
|
#include "../jrd/err_proto.h"
|
2009-08-21 10:36:37 +02:00
|
|
|
#include "../jrd/isc_proto.h"
|
2006-05-31 10:53:00 +02:00
|
|
|
#include "../jrd/os/path_utils.h"
|
|
|
|
|
|
|
|
#include "../jrd/TempSpace.h"
|
|
|
|
|
2009-06-06 20:39:29 +02:00
|
|
|
using Firebird::TempFile;
|
|
|
|
|
2006-05-31 10:53:00 +02:00
|
|
|
// Static definitions/initializations
|
|
|
|
|
2010-01-21 03:51:32 +01:00
|
|
|
const size_t MIN_TEMP_BLOCK_SIZE = 64 * 1024;
|
2010-01-19 09:25:42 +01:00
|
|
|
|
2008-01-23 16:52:40 +01:00
|
|
|
Firebird::GlobalPtr<Firebird::Mutex> TempSpace::initMutex;
|
2006-05-31 10:53:00 +02:00
|
|
|
Firebird::TempDirectoryList* TempSpace::tempDirs = NULL;
|
|
|
|
size_t TempSpace::minBlockSize = 0;
|
|
|
|
offset_t TempSpace::globalCacheUsage = 0;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Generic space block class
|
|
|
|
//
|
|
|
|
|
|
|
|
TempSpace::Block::Block(Block* tail, size_t length)
|
|
|
|
: next(NULL), size(length)
|
|
|
|
{
|
|
|
|
if (tail)
|
|
|
|
{
|
|
|
|
tail->next = this;
|
|
|
|
}
|
|
|
|
prev = tail;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// In-memory block class
|
|
|
|
//
|
|
|
|
|
|
|
|
TempSpace::MemoryBlock::MemoryBlock(MemoryPool& pool, Block* tail, size_t length)
|
|
|
|
: Block(tail, length)
|
|
|
|
{
|
2010-03-19 11:54:53 +01:00
|
|
|
ptr = FB_NEW(pool) UCHAR[length];
|
2006-05-31 10:53:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
TempSpace::MemoryBlock::~MemoryBlock()
|
|
|
|
{
|
|
|
|
delete[] ptr;
|
|
|
|
}
|
|
|
|
|
2007-03-12 10:39:52 +01:00
|
|
|
size_t TempSpace::MemoryBlock::read(offset_t offset, void* buffer, size_t length)
|
2006-05-31 10:53:00 +02:00
|
|
|
{
|
|
|
|
if (offset + length > size)
|
|
|
|
{
|
|
|
|
length = size - offset;
|
|
|
|
}
|
2006-06-29 11:06:32 +02:00
|
|
|
memcpy(buffer, ptr + offset, length);
|
2006-05-31 10:53:00 +02:00
|
|
|
return length;
|
|
|
|
}
|
|
|
|
|
2008-04-19 11:42:01 +02:00
|
|
|
size_t TempSpace::MemoryBlock::write(offset_t offset, const void* buffer, size_t length)
|
2006-05-31 10:53:00 +02:00
|
|
|
{
|
|
|
|
if (offset + length > size)
|
|
|
|
{
|
|
|
|
length = size - offset;
|
|
|
|
}
|
2006-06-29 11:06:32 +02:00
|
|
|
memcpy(ptr + offset, buffer, length);
|
2006-05-31 10:53:00 +02:00
|
|
|
return length;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// On-disk block class
|
|
|
|
//
|
|
|
|
|
|
|
|
TempSpace::FileBlock::FileBlock(TempFile* f, Block* tail, size_t length)
|
|
|
|
: Block(tail, length), file(f)
|
|
|
|
{
|
|
|
|
fb_assert(file);
|
2008-12-05 02:20:14 +01:00
|
|
|
|
2006-08-29 23:48:15 +02:00
|
|
|
// FileBlock is created after file was extended by length (look at
|
|
|
|
// TempSpace::extend) so this FileBlock is already inside the file
|
|
|
|
seek = file->getSize() - length;
|
2006-05-31 10:53:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
TempSpace::FileBlock::~FileBlock()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2007-03-12 10:39:52 +01:00
|
|
|
size_t TempSpace::FileBlock::read(offset_t offset, void* buffer, size_t length)
|
2006-05-31 10:53:00 +02:00
|
|
|
{
|
|
|
|
if (offset + length > size)
|
|
|
|
{
|
|
|
|
length = size - offset;
|
|
|
|
}
|
|
|
|
offset += seek;
|
|
|
|
return file->read(offset, buffer, length);
|
|
|
|
}
|
|
|
|
|
2008-04-19 11:42:01 +02:00
|
|
|
size_t TempSpace::FileBlock::write(offset_t offset, const void* buffer, size_t length)
|
2006-05-31 10:53:00 +02:00
|
|
|
{
|
|
|
|
if (offset + length > size)
|
|
|
|
{
|
|
|
|
length = size - offset;
|
|
|
|
}
|
|
|
|
offset += seek;
|
|
|
|
return file->write(offset, buffer, length);
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// TempSpace::TempSpace
|
|
|
|
//
|
|
|
|
// Constructor
|
|
|
|
//
|
|
|
|
|
2010-03-03 16:02:01 +01:00
|
|
|
TempSpace::TempSpace(MemoryPool& p, const Firebird::PathName& prefix)
|
|
|
|
: pool(p), filePrefix(p, prefix),
|
2010-03-01 03:14:36 +01:00
|
|
|
logicalSize(0), physicalSize(0), localCacheUsage(0),
|
|
|
|
head(NULL), tail(NULL), tempFiles(p),
|
|
|
|
freeSegments(NULL), notUsedSegments(NULL)
|
2006-05-31 10:53:00 +02:00
|
|
|
{
|
|
|
|
if (!tempDirs)
|
|
|
|
{
|
|
|
|
Firebird::MutexLockGuard guard(initMutex);
|
|
|
|
if (!tempDirs)
|
|
|
|
{
|
2006-06-29 11:06:32 +02:00
|
|
|
MemoryPool& def_pool = *getDefaultMemoryPool();
|
|
|
|
tempDirs = FB_NEW(def_pool) Firebird::TempDirectoryList(def_pool);
|
2010-03-03 16:02:01 +01:00
|
|
|
minBlockSize = Config::getTempBlockSize();
|
2010-01-19 09:25:42 +01:00
|
|
|
|
|
|
|
if (minBlockSize < MIN_TEMP_BLOCK_SIZE)
|
|
|
|
minBlockSize = MIN_TEMP_BLOCK_SIZE;
|
|
|
|
else
|
|
|
|
minBlockSize = FB_ALIGN(minBlockSize, MIN_TEMP_BLOCK_SIZE);
|
2006-05-31 10:53:00 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// TempSpace::~TempSpace
|
|
|
|
//
|
|
|
|
// Destructor
|
|
|
|
//
|
|
|
|
|
|
|
|
TempSpace::~TempSpace()
|
|
|
|
{
|
|
|
|
while (head)
|
|
|
|
{
|
|
|
|
Block* temp = head->next;
|
|
|
|
delete head;
|
|
|
|
head = temp;
|
|
|
|
}
|
|
|
|
|
|
|
|
globalCacheUsage -= localCacheUsage;
|
|
|
|
|
|
|
|
while (tempFiles.getCount())
|
|
|
|
{
|
|
|
|
delete tempFiles.pop();
|
|
|
|
}
|
2007-04-03 14:57:32 +02:00
|
|
|
|
|
|
|
while (freeSegments)
|
|
|
|
{
|
|
|
|
Segment* temp = freeSegments->next;
|
|
|
|
delete freeSegments;
|
|
|
|
freeSegments = temp;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (notUsedSegments)
|
|
|
|
{
|
|
|
|
Segment* temp = notUsedSegments->next;
|
|
|
|
delete notUsedSegments;
|
|
|
|
notUsedSegments = temp;
|
|
|
|
}
|
2006-05-31 10:53:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// TempSpace::read
|
|
|
|
//
|
|
|
|
// Reads bytes from the temporary space
|
|
|
|
//
|
|
|
|
|
|
|
|
size_t TempSpace::read(offset_t offset, void* buffer, size_t length)
|
|
|
|
{
|
|
|
|
fb_assert(offset + length <= logicalSize);
|
|
|
|
|
|
|
|
if (length)
|
|
|
|
{
|
|
|
|
// search for the first needed block
|
|
|
|
Block* block = findBlock(offset);
|
|
|
|
|
2010-03-19 11:54:53 +01:00
|
|
|
UCHAR* p = static_cast<UCHAR*>(buffer);
|
2006-05-31 10:53:00 +02:00
|
|
|
size_t l = length;
|
|
|
|
|
|
|
|
// read data from the block chain
|
2008-04-19 11:42:01 +02:00
|
|
|
for (Block* itr = block; itr && l; itr = itr->next, offset = 0)
|
2006-05-31 10:53:00 +02:00
|
|
|
{
|
|
|
|
const size_t n = itr->read(offset, p, l);
|
|
|
|
p += n;
|
|
|
|
l -= n;
|
|
|
|
}
|
|
|
|
|
|
|
|
fb_assert(!l);
|
|
|
|
}
|
|
|
|
|
|
|
|
return length;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// TempSpace::write
|
|
|
|
//
|
|
|
|
// Writes bytes to the temporary space
|
|
|
|
//
|
|
|
|
|
2008-04-19 11:42:01 +02:00
|
|
|
size_t TempSpace::write(offset_t offset, const void* buffer, size_t length)
|
2006-05-31 10:53:00 +02:00
|
|
|
{
|
|
|
|
fb_assert(offset <= logicalSize);
|
|
|
|
|
|
|
|
if (offset + length > logicalSize)
|
|
|
|
{
|
|
|
|
// not enough space, allocate one more block
|
|
|
|
extend(offset + length - logicalSize);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (length)
|
|
|
|
{
|
|
|
|
// search for the first needed block
|
2008-04-19 11:42:01 +02:00
|
|
|
Block* const block = findBlock(offset);
|
2006-05-31 10:53:00 +02:00
|
|
|
|
2010-03-19 11:54:53 +01:00
|
|
|
const UCHAR* p = static_cast<const UCHAR*>(buffer);
|
2006-05-31 10:53:00 +02:00
|
|
|
size_t l = length;
|
|
|
|
|
|
|
|
// write data to as many blocks as necessary
|
2008-04-19 11:42:01 +02:00
|
|
|
for (Block* itr = block; itr && l; itr = itr->next, offset = 0)
|
2006-05-31 10:53:00 +02:00
|
|
|
{
|
|
|
|
const size_t n = itr->write(offset, p, l);
|
|
|
|
p += n;
|
|
|
|
l -= n;
|
|
|
|
}
|
|
|
|
|
|
|
|
fb_assert(!l);
|
|
|
|
}
|
|
|
|
|
|
|
|
return length;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// TempSpace::extend
|
|
|
|
//
|
|
|
|
// Increases size of the temporary space
|
|
|
|
//
|
|
|
|
|
|
|
|
void TempSpace::extend(size_t size)
|
|
|
|
{
|
|
|
|
logicalSize += size;
|
|
|
|
|
|
|
|
if (logicalSize > physicalSize)
|
|
|
|
{
|
2006-08-29 09:09:00 +02:00
|
|
|
size = FB_ALIGN(logicalSize - physicalSize, minBlockSize);
|
2006-05-31 10:53:00 +02:00
|
|
|
physicalSize += size;
|
|
|
|
|
|
|
|
Block* block = NULL;
|
|
|
|
|
2010-03-03 16:02:01 +01:00
|
|
|
if (globalCacheUsage + size <= size_t(Config::getTempCacheLimit()))
|
2006-05-31 10:53:00 +02:00
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
// allocate block in virtual memory
|
|
|
|
block = FB_NEW(pool) MemoryBlock(pool, tail, size);
|
|
|
|
localCacheUsage += size;
|
|
|
|
globalCacheUsage += size;
|
|
|
|
}
|
|
|
|
catch (const Firebird::BadAlloc&)
|
|
|
|
{
|
|
|
|
// not enough memory
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!block)
|
|
|
|
{
|
|
|
|
// allocate block in the temp file
|
2009-08-21 10:36:37 +02:00
|
|
|
TempFile* const file = setupFile(size);
|
|
|
|
fb_assert(file);
|
2007-04-03 14:57:32 +02:00
|
|
|
if (tail && tail->sameFile(file))
|
|
|
|
{
|
|
|
|
tail->size += size;
|
|
|
|
return;
|
|
|
|
}
|
2006-05-31 10:53:00 +02:00
|
|
|
block = FB_NEW(pool) FileBlock(file, tail, size);
|
|
|
|
}
|
|
|
|
|
|
|
|
// append new block to the chain
|
|
|
|
if (!head)
|
|
|
|
{
|
|
|
|
head = block;
|
|
|
|
}
|
|
|
|
tail = block;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// TempSpace::findBlock
|
|
|
|
//
|
|
|
|
// Locates the space block corresponding to the given global offset
|
|
|
|
//
|
|
|
|
|
2007-04-03 14:57:32 +02:00
|
|
|
TempSpace::Block* TempSpace::findBlock(offset_t& offset) const
|
2006-05-31 10:53:00 +02:00
|
|
|
{
|
|
|
|
fb_assert(offset <= logicalSize);
|
|
|
|
|
|
|
|
Block* block = NULL;
|
|
|
|
|
|
|
|
if (offset < physicalSize / 2)
|
|
|
|
{
|
|
|
|
// walk forward
|
|
|
|
block = head;
|
|
|
|
while (block && offset >= block->size)
|
|
|
|
{
|
|
|
|
offset -= block->size;
|
|
|
|
block = block->next;
|
|
|
|
}
|
|
|
|
fb_assert(block);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// walk backward
|
|
|
|
block = tail;
|
|
|
|
while (block && physicalSize - offset > block->size)
|
|
|
|
{
|
|
|
|
offset += block->size;
|
|
|
|
block = block->prev;
|
|
|
|
}
|
|
|
|
fb_assert(block);
|
|
|
|
offset -= physicalSize - block->size;
|
|
|
|
}
|
|
|
|
|
|
|
|
fb_assert(offset <= block->size);
|
|
|
|
return block;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// TempSpace::setupFile
|
|
|
|
//
|
|
|
|
// Allocates the required space in some temporary file
|
|
|
|
//
|
|
|
|
|
|
|
|
TempFile* TempSpace::setupFile(size_t size)
|
|
|
|
{
|
2009-08-23 03:27:46 +02:00
|
|
|
ISC_STATUS_ARRAY status_vector = {0};
|
2006-05-31 10:53:00 +02:00
|
|
|
|
|
|
|
for (size_t i = 0; i < tempDirs->getCount(); i++)
|
|
|
|
{
|
2009-08-21 10:36:37 +02:00
|
|
|
TempFile* file = NULL;
|
|
|
|
|
2006-05-31 10:53:00 +02:00
|
|
|
Firebird::PathName directory = (*tempDirs)[i];
|
2006-06-03 03:01:51 +02:00
|
|
|
PathUtils::ensureSeparator(directory);
|
2006-05-31 10:53:00 +02:00
|
|
|
|
|
|
|
for (size_t j = 0; j < tempFiles.getCount(); j++)
|
|
|
|
{
|
|
|
|
Firebird::PathName dirname, filename;
|
2008-09-13 13:09:09 +02:00
|
|
|
PathUtils::splitLastComponent(dirname, filename, tempFiles[j]->getName());
|
2006-06-02 10:46:28 +02:00
|
|
|
PathUtils::ensureSeparator(dirname);
|
2006-06-05 16:39:33 +02:00
|
|
|
if (!directory.compare(dirname))
|
2006-05-31 10:53:00 +02:00
|
|
|
{
|
2008-02-06 20:27:57 +01:00
|
|
|
file = tempFiles[j];
|
2006-05-31 10:53:00 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
2008-02-06 20:27:57 +01:00
|
|
|
if (!file)
|
|
|
|
{
|
|
|
|
file = FB_NEW(pool) TempFile(pool, filePrefix, directory);
|
|
|
|
tempFiles.add(file);
|
|
|
|
}
|
|
|
|
|
2006-05-31 10:53:00 +02:00
|
|
|
file->extend(size);
|
|
|
|
}
|
2009-08-21 10:36:37 +02:00
|
|
|
catch (const Firebird::system_error& ex)
|
2006-05-31 10:53:00 +02:00
|
|
|
{
|
2009-08-21 10:36:37 +02:00
|
|
|
ex.stuff_exception(status_vector);
|
|
|
|
continue;
|
2006-05-31 10:53:00 +02:00
|
|
|
}
|
|
|
|
|
2009-08-21 10:36:37 +02:00
|
|
|
return file;
|
2006-05-31 10:53:00 +02:00
|
|
|
}
|
|
|
|
|
2009-08-21 10:36:37 +02:00
|
|
|
// no room in all directories
|
|
|
|
Firebird::Arg::Gds status(isc_out_of_temp_space);
|
|
|
|
status.append(Firebird::Arg::StatusVector(status_vector));
|
|
|
|
iscLogStatus(NULL, status.value());
|
|
|
|
status.raise();
|
|
|
|
|
|
|
|
return NULL; // compiler silencer
|
2006-05-31 10:53:00 +02:00
|
|
|
}
|
2007-04-03 14:57:32 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// TempSpace::allocateSpace
|
|
|
|
//
|
|
|
|
// Allocate available space in free segments. Extend file if necessary
|
|
|
|
//
|
|
|
|
|
|
|
|
offset_t TempSpace::allocateSpace(size_t size)
|
|
|
|
{
|
|
|
|
// Find the best available space. This is defined as the smallest free space
|
|
|
|
// that is big enough. This preserves large blocks.
|
|
|
|
Segment** best = NULL, *space;
|
|
|
|
|
2007-04-05 12:28:11 +02:00
|
|
|
// Search through the available space in the not used segments list
|
2008-09-13 13:09:09 +02:00
|
|
|
for (Segment** ptr = &freeSegments; (space = *ptr); ptr = &(*ptr)->next)
|
2007-04-03 14:57:32 +02:00
|
|
|
{
|
|
|
|
// If this is smaller than our previous best, use it
|
|
|
|
if (space->size >= size && (!best || (space->size < (*best)->size))) {
|
|
|
|
best = ptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we didn't find any space, allocate it at the end of the file
|
2008-12-05 02:20:14 +01:00
|
|
|
if (!best)
|
2007-04-03 14:57:32 +02:00
|
|
|
{
|
|
|
|
extend(size);
|
|
|
|
return getSize() - size;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set up the return parameters
|
|
|
|
space = *best;
|
|
|
|
|
|
|
|
// If the hunk was an exact fit, remove the segment from the
|
2007-04-05 12:28:11 +02:00
|
|
|
// list and splice it into the not used segments list
|
2008-12-05 02:20:14 +01:00
|
|
|
if (space->size == size)
|
2007-04-03 14:57:32 +02:00
|
|
|
{
|
|
|
|
*best = space->next;
|
|
|
|
space->next = notUsedSegments;
|
|
|
|
notUsedSegments = space;
|
|
|
|
return space->position;
|
|
|
|
}
|
|
|
|
|
|
|
|
// The best segment is too big - chop the needed space off the begin
|
|
|
|
space->size -= size;
|
|
|
|
space->position += size;
|
|
|
|
return (space->position - size);
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// TempSpace::releaseSpace
|
|
|
|
//
|
2008-12-05 02:20:14 +01:00
|
|
|
// Return previously allocated segment back into not used segments list and
|
2007-04-03 14:57:32 +02:00
|
|
|
// join it with adjacent segments if found
|
|
|
|
//
|
|
|
|
|
|
|
|
void TempSpace::releaseSpace(offset_t position, size_t size)
|
|
|
|
{
|
|
|
|
fb_assert(size > 0);
|
|
|
|
fb_assert(position < getSize()); // Block starts in file
|
|
|
|
const offset_t end = position + size;
|
|
|
|
fb_assert(end <= getSize()); // Block ends in file
|
|
|
|
|
|
|
|
Segment* new_seg = NULL;
|
|
|
|
Segment* space = freeSegments;
|
|
|
|
if (!space || end < space->position)
|
|
|
|
{
|
|
|
|
new_seg = getSegment(position, size);
|
|
|
|
freeSegments = new_seg;
|
|
|
|
new_seg->next = space;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (end == space->position || position == space->position + space->size)
|
|
|
|
{
|
|
|
|
joinSegment(space, position, size);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (true)
|
|
|
|
{
|
|
|
|
Segment* next = space->next;
|
|
|
|
if (!next || end < next->position)
|
|
|
|
{
|
|
|
|
new_seg = getSegment(position, size);
|
|
|
|
space->next = new_seg;
|
|
|
|
new_seg->next = next;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (end == next->position || position == next->position + next->size)
|
|
|
|
{
|
|
|
|
joinSegment(next, position, size);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
space = next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// TempSpace::inMemory
|
|
|
|
//
|
|
|
|
// Return contiguous chunk of memory if present at given location
|
|
|
|
//
|
|
|
|
|
2010-03-19 11:54:53 +01:00
|
|
|
UCHAR* TempSpace::inMemory(offset_t begin, size_t size) const
|
2007-04-03 14:57:32 +02:00
|
|
|
{
|
2007-04-05 11:13:10 +02:00
|
|
|
const Block* block = findBlock(begin);
|
2008-01-16 11:25:04 +01:00
|
|
|
return block ? block->inMemory(begin, size) : NULL;
|
2007-04-03 14:57:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// TempSpace::findMemory
|
|
|
|
//
|
2008-12-05 02:20:14 +01:00
|
|
|
// Return contiguous chunk of memory and adjust starting offset
|
2007-04-03 14:57:32 +02:00
|
|
|
// of search range if found
|
|
|
|
//
|
|
|
|
|
2010-03-19 11:54:53 +01:00
|
|
|
UCHAR* TempSpace::findMemory(offset_t& begin, offset_t end, size_t size) const
|
2007-04-03 14:57:32 +02:00
|
|
|
{
|
|
|
|
offset_t local_offset = begin;
|
|
|
|
const offset_t save_begin = begin;
|
2007-04-05 11:13:10 +02:00
|
|
|
const Block* block = findBlock(local_offset);
|
2008-01-16 11:25:04 +01:00
|
|
|
|
2007-04-03 14:57:32 +02:00
|
|
|
while (block && (begin + size <= end))
|
|
|
|
{
|
2010-03-19 11:54:53 +01:00
|
|
|
UCHAR* mem = block->inMemory(local_offset, size);
|
2008-12-05 02:20:14 +01:00
|
|
|
if (mem)
|
2007-04-03 14:57:32 +02:00
|
|
|
{
|
|
|
|
return mem;
|
|
|
|
}
|
2008-01-16 11:25:04 +01:00
|
|
|
|
|
|
|
begin += block->size - local_offset;
|
|
|
|
local_offset = 0;
|
|
|
|
block = block->next;
|
2007-04-03 14:57:32 +02:00
|
|
|
}
|
2008-01-16 11:25:04 +01:00
|
|
|
|
2007-04-03 14:57:32 +02:00
|
|
|
begin = save_begin;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// TempSpace::validate
|
|
|
|
//
|
|
|
|
// Validate internal lists for consistency and return back to caller
|
|
|
|
// amount of available free space
|
|
|
|
//
|
|
|
|
|
|
|
|
bool TempSpace::validate(offset_t& free) const
|
|
|
|
{
|
|
|
|
free = 0;
|
2007-04-05 11:13:10 +02:00
|
|
|
for (const Segment* space = freeSegments; space; space = space->next)
|
2007-04-03 14:57:32 +02:00
|
|
|
{
|
2007-04-05 11:13:10 +02:00
|
|
|
free += space->size;
|
2009-01-03 20:02:04 +01:00
|
|
|
fb_assert(!(space->next) || (space->next->position > space->position));
|
2007-04-03 14:57:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
offset_t disk = 0;
|
2007-04-11 11:28:50 +02:00
|
|
|
for (size_t i = 0; i < tempFiles.getCount(); i++)
|
2007-04-03 14:57:32 +02:00
|
|
|
disk += tempFiles[i]->getSize();
|
|
|
|
|
|
|
|
return ((localCacheUsage + disk) == physicalSize);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// TempSpace::allocateBatch
|
|
|
|
//
|
2008-12-05 02:20:14 +01:00
|
|
|
// Allocate up to 'count' contiguous chunks of memory available in free
|
2007-04-03 14:57:32 +02:00
|
|
|
// segments if any. Adjust size of chunks between minSize and maxSize
|
2008-12-05 02:20:14 +01:00
|
|
|
// accordingly to available free space (assuming all of the free space
|
2007-04-03 14:57:32 +02:00
|
|
|
// is in memory blocks). Algorithm is very simple and can be improved in future
|
2008-12-05 02:20:14 +01:00
|
|
|
//
|
2007-04-03 14:57:32 +02:00
|
|
|
|
|
|
|
size_t TempSpace::allocateBatch(size_t count, size_t minSize, size_t maxSize, Segments& segments)
|
|
|
|
{
|
|
|
|
// adjust passed chunk size to amount of free memory we have and number
|
2008-12-05 02:20:14 +01:00
|
|
|
// of runs still not allocated.
|
2007-04-03 14:57:32 +02:00
|
|
|
offset_t freeMem = 0;
|
|
|
|
Segment* freeSpace = freeSegments;
|
|
|
|
for (; freeSpace; freeSpace = freeSpace->next)
|
|
|
|
freeMem += freeSpace->size;
|
|
|
|
|
|
|
|
freeMem = MIN(freeMem / count, maxSize);
|
|
|
|
freeMem = MAX(freeMem, minSize);
|
|
|
|
freeMem = MIN(freeMem, minBlockSize);
|
2010-01-19 09:25:42 +01:00
|
|
|
freeMem &= ~(FB_ALIGNMENT - 1);
|
2008-12-05 02:20:14 +01:00
|
|
|
|
2007-04-03 14:57:32 +02:00
|
|
|
Segment** prevSpace = &freeSegments;
|
|
|
|
freeSpace = freeSegments;
|
|
|
|
offset_t freeSeek = freeSpace ? freeSpace->position : 0;
|
|
|
|
offset_t freeEnd = freeSpace ? freeSpace->position + freeSpace->size : 0;
|
|
|
|
while (segments.getCount() < count && freeSpace)
|
|
|
|
{
|
2010-03-19 11:54:53 +01:00
|
|
|
UCHAR* mem = findMemory(freeSeek, freeEnd, freeMem);
|
2007-04-03 14:57:32 +02:00
|
|
|
|
|
|
|
if (mem)
|
|
|
|
{
|
|
|
|
fb_assert(freeSeek + freeMem <= freeEnd);
|
|
|
|
#ifdef DEV_BUILD
|
|
|
|
offset_t seek1 = freeSeek;
|
2010-03-19 11:54:53 +01:00
|
|
|
UCHAR* p = findMemory(seek1, freeEnd, freeMem);
|
2007-04-03 14:57:32 +02:00
|
|
|
fb_assert(p == mem);
|
|
|
|
fb_assert(seek1 == freeSeek);
|
|
|
|
#endif
|
|
|
|
if (freeSeek != freeSpace->position)
|
|
|
|
{
|
|
|
|
const ULONG skip_size = freeSeek - freeSpace->position;
|
|
|
|
Segment* skip_space = getSegment(freeSpace->position, skip_size);
|
|
|
|
|
|
|
|
(*prevSpace) = skip_space;
|
|
|
|
skip_space->next = freeSpace;
|
|
|
|
prevSpace = &skip_space->next;
|
|
|
|
|
|
|
|
freeSpace->position += skip_size;
|
|
|
|
freeSpace->size -= skip_size;
|
|
|
|
}
|
|
|
|
|
|
|
|
SegmentInMemory seg;
|
|
|
|
seg.memory = mem;
|
|
|
|
seg.position = freeSeek;
|
|
|
|
seg.size = freeMem;
|
|
|
|
segments.add(seg);
|
|
|
|
|
|
|
|
freeSeek += freeMem;
|
|
|
|
freeSpace->position += freeMem;
|
|
|
|
freeSpace->size -= freeMem;
|
|
|
|
if (!freeSpace->size)
|
|
|
|
{
|
|
|
|
(*prevSpace) = freeSpace->next;
|
|
|
|
freeSpace->next = notUsedSegments;
|
|
|
|
notUsedSegments = freeSpace;
|
|
|
|
|
|
|
|
freeSpace = (*prevSpace);
|
|
|
|
freeSeek = freeSpace ? freeSpace->position : 0;
|
|
|
|
freeEnd = freeSpace ? freeSpace->position + freeSpace->size : 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
prevSpace = &freeSpace->next;
|
|
|
|
freeSpace = freeSpace->next;
|
|
|
|
freeSeek = freeSpace ? freeSpace->position : 0;
|
|
|
|
freeEnd = freeSpace ? freeSpace->position + freeSpace->size : 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return segments.getCount();
|
|
|
|
}
|
2007-04-04 23:34:51 +02:00
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// TempSpace::getSegment
|
|
|
|
//
|
|
|
|
// Return not used Segment instance or allocate new one
|
2008-12-05 02:20:14 +01:00
|
|
|
//
|
2007-04-04 23:34:51 +02:00
|
|
|
|
|
|
|
TempSpace::Segment* TempSpace::getSegment(offset_t position, size_t size)
|
|
|
|
{
|
|
|
|
Segment* result = notUsedSegments;
|
|
|
|
|
2008-12-05 02:20:14 +01:00
|
|
|
if (result)
|
2007-04-04 23:34:51 +02:00
|
|
|
{
|
|
|
|
notUsedSegments = result->next;
|
|
|
|
|
|
|
|
result->next = NULL;
|
|
|
|
result->position = position;
|
|
|
|
result->size = size;
|
|
|
|
}
|
2008-12-05 02:20:14 +01:00
|
|
|
else
|
2007-04-04 23:34:51 +02:00
|
|
|
{
|
|
|
|
result = (Segment*) FB_NEW(pool) Segment(NULL, position, size);
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// TempSpace::joinSegment
|
|
|
|
//
|
2008-12-05 02:20:14 +01:00
|
|
|
// Extend existing segment and join it with adjacent segment
|
|
|
|
//
|
2007-04-04 23:34:51 +02:00
|
|
|
|
|
|
|
void TempSpace::joinSegment(Segment* seg, offset_t position, size_t size)
|
|
|
|
{
|
|
|
|
if (position + size == seg->position)
|
|
|
|
{
|
|
|
|
seg->position -= size;
|
|
|
|
seg->size += size;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
seg->size += size;
|
|
|
|
Segment* next = seg->next;
|
|
|
|
if (next && next->position == seg->position + seg->size)
|
|
|
|
{
|
|
|
|
seg->next = next->next;
|
|
|
|
seg->size += next->size;
|
|
|
|
|
|
|
|
next->next = notUsedSegments;
|
|
|
|
notUsedSegments = next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|