8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-25 04:43:03 +01:00
firebird-mirror/src/jrd/TempSpace.cpp

665 lines
15 KiB
C++
Raw Normal View History

/*
* 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 "iberror.h"
#include "../common/config/config.h"
#include "../common/config/dir_list.h"
#include "../common/gdsassert.h"
2010-10-12 10:02:57 +02:00
#include "../yvalve/gds_proto.h"
#include "../jrd/err_proto.h"
2010-10-12 10:02:57 +02:00
#include "../common/isc_proto.h"
#include "../common/os/path_utils.h"
#include "../jrd/TempSpace.h"
2009-06-06 20:39:29 +02:00
using Firebird::TempFile;
// Static definitions/initializations
2012-03-17 03:26:59 +01:00
const size_t MIN_TEMP_BLOCK_SIZE = 64 * 1024;
Firebird::GlobalPtr<Firebird::Mutex> TempSpace::initMutex;
Firebird::TempDirectoryList* TempSpace::tempDirs = NULL;
2014-07-17 20:48:46 +02:00
FB_SIZE_T TempSpace::minBlockSize = 0;
offset_t TempSpace::globalCacheUsage = 0;
//
// In-memory block class
//
2014-07-17 20:48:46 +02:00
FB_SIZE_T TempSpace::MemoryBlock::read(offset_t offset, void* buffer, FB_SIZE_T length)
{
if (offset + length > size)
{
length = size - offset;
}
2006-06-29 11:06:32 +02:00
memcpy(buffer, ptr + offset, length);
return length;
}
2014-07-17 20:48:46 +02:00
FB_SIZE_T TempSpace::MemoryBlock::write(offset_t offset, const void* buffer, FB_SIZE_T length)
{
if (offset + length > size)
{
length = size - offset;
}
2006-06-29 11:06:32 +02:00
memcpy(ptr + offset, buffer, length);
return length;
}
//
// On-disk block class
//
2014-07-17 20:48:46 +02:00
FB_SIZE_T TempSpace::FileBlock::read(offset_t offset, void* buffer, FB_SIZE_T length)
{
if (offset + length > size)
{
length = size - offset;
}
offset += seek;
return file->read(offset, buffer, length);
}
2014-07-17 20:48:46 +02:00
FB_SIZE_T TempSpace::FileBlock::write(offset_t offset, const void* buffer, FB_SIZE_T length)
{
if (offset + length > size)
{
length = size - offset;
}
offset += seek;
return file->write(offset, buffer, length);
}
//
// TempSpace::TempSpace
//
// Constructor
//
TempSpace::TempSpace(MemoryPool& p, const Firebird::PathName& prefix, bool dynamic)
: pool(p), filePrefix(p, prefix),
logicalSize(0), physicalSize(0), localCacheUsage(0),
head(NULL), tail(NULL), tempFiles(p),
initialBuffer(p), initiallyDynamic(dynamic),
freeSegments(p)
{
if (!tempDirs)
{
Firebird::MutexLockGuard guard(initMutex, FB_FUNCTION);
if (!tempDirs)
{
2006-06-29 11:06:32 +02:00
MemoryPool& def_pool = *getDefaultMemoryPool();
tempDirs = FB_NEW(def_pool) Firebird::TempDirectoryList(def_pool);
minBlockSize = Config::getTempBlockSize();
if (minBlockSize < MIN_TEMP_BLOCK_SIZE)
minBlockSize = MIN_TEMP_BLOCK_SIZE;
else
minBlockSize = fbAlign(minBlockSize, MIN_TEMP_BLOCK_SIZE);
}
}
}
//
// TempSpace::~TempSpace
//
// Destructor
//
TempSpace::~TempSpace()
{
while (head)
{
Block* temp = head->next;
delete head;
head = temp;
}
globalCacheUsage -= localCacheUsage;
while (tempFiles.getCount())
{
delete tempFiles.pop();
}
}
//
// TempSpace::read
//
// Reads bytes from the temporary space
//
2014-07-17 20:48:46 +02:00
FB_SIZE_T TempSpace::read(offset_t offset, void* buffer, FB_SIZE_T length)
{
fb_assert(offset + length <= logicalSize);
if (length)
{
// search for the first needed block
Block* block = findBlock(offset);
UCHAR* p = static_cast<UCHAR*>(buffer);
2014-07-17 20:48:46 +02:00
FB_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)
{
2014-07-17 20:48:46 +02:00
const FB_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
//
2014-07-17 20:48:46 +02:00
FB_SIZE_T TempSpace::write(offset_t offset, const void* buffer, FB_SIZE_T length)
{
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);
const UCHAR* p = static_cast<const UCHAR*>(buffer);
2014-07-17 20:48:46 +02:00
FB_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)
{
2014-07-17 20:48:46 +02:00
const FB_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
//
2014-07-17 20:48:46 +02:00
void TempSpace::extend(FB_SIZE_T size)
{
logicalSize += size;
if (logicalSize > physicalSize)
{
2014-07-17 20:48:46 +02:00
const FB_SIZE_T initialSize = initialBuffer.getCount();
// If the dynamic mode is specified, then we allocate new blocks
// by growing the same initial memory block in the specified chunks.
// Once the limit (64KB) is reached, we switch to the generic algorithm
// (1MB blocks), copy the existing data there and free the initial buffer.
//
// This mode should not be used if the caller never works with small blocks.
// Also, it MUST NOT be used if the caller deals with inMemory() or allocateBatch()
// routines and caches the pointers to use them later. These pointers may become
// invalid after resizing the initial block or after switching to large blocks.
if (initiallyDynamic && logicalSize < MIN_TEMP_BLOCK_SIZE)
{
// allocate or extend the initial dynamic block, it will grow up to 64KB
if (!initialSize)
{
fb_assert(!head && !tail);
head = tail = FB_NEW(pool) InitialBlock(initialBuffer.getBuffer(size), size);
}
else
{
fb_assert(head == tail);
size += initialSize;
initialBuffer.resize(size);
new (head) InitialBlock(initialBuffer.begin(), size);
}
physicalSize = size;
return;
}
if (initialSize)
{
fb_assert(head == tail);
delete head;
head = tail = NULL;
size = fbAlign(logicalSize, minBlockSize);
physicalSize = size;
}
else
{
size = fbAlign(logicalSize - physicalSize, minBlockSize);
physicalSize += size;
}
Block* block = NULL;
if (globalCacheUsage + size <= size_t(Config::getTempCacheLimit()))
{
try
{
// allocate block in virtual memory
block = FB_NEW(pool) MemoryBlock(FB_NEW(pool) UCHAR[size], tail, size);
localCacheUsage += size;
globalCacheUsage += size;
}
catch (const Firebird::BadAlloc&)
{
// not enough memory
}
}
if (!block)
{
// allocate block in the temp file
TempFile* const file = setupFile(size);
fb_assert(file);
if (tail && tail->sameFile(file))
{
fb_assert(!initialSize);
tail->size += size;
return;
}
block = FB_NEW(pool) FileBlock(file, tail, size);
}
// preserve the initial contents, if any
if (initialSize)
{
block->write(0, initialBuffer.begin(), initialSize);
initialBuffer.free();
}
// append new block to the chain
if (!head)
{
head = block;
}
tail = block;
}
}
//
// TempSpace::findBlock
//
// Locates the space block corresponding to the given global offset
//
TempSpace::Block* TempSpace::findBlock(offset_t& offset) const
{
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
//
2014-07-17 20:48:46 +02:00
TempFile* TempSpace::setupFile(FB_SIZE_T size)
{
2009-08-23 03:27:46 +02:00
ISC_STATUS_ARRAY status_vector = {0};
2014-07-17 20:48:46 +02:00
for (FB_SIZE_T i = 0; i < tempDirs->getCount(); i++)
{
TempFile* file = NULL;
Firebird::PathName directory = (*tempDirs)[i];
2006-06-03 03:01:51 +02:00
PathUtils::ensureSeparator(directory);
2014-07-17 20:48:46 +02:00
for (FB_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());
PathUtils::ensureSeparator(dirname);
2006-06-05 16:39:33 +02:00
if (!directory.compare(dirname))
{
file = tempFiles[j];
break;
}
}
try
{
if (!file)
{
file = FB_NEW(pool) TempFile(pool, filePrefix, directory);
tempFiles.add(file);
}
file->extend(size);
}
catch (const Firebird::system_error& ex)
{
ex.stuff_exception(status_vector);
continue;
}
return file;
}
// 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
}
//
// TempSpace::allocateSpace
//
// Allocate available space in free segments. Extend file if necessary
//
2014-07-17 20:48:46 +02:00
offset_t TempSpace::allocateSpace(FB_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;
// Search through the available space in the not used segments list
for (bool found = freeSegments.getFirst(); found; found = freeSegments.getNext())
{
Segment* const space = &freeSegments.current();
// If this is smaller than our previous best, use it
if (space->size >= size && (!best || (space->size < best->size))) {
best = space;
}
}
// 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)
{
extend(size);
return getSize() - size;
}
// Set up the return parameters
const offset_t position = best->position;
best->size -= size;
best->position += size;
// If the hunk was an exact fit, remove the segment from the list
if (!best->size)
{
if (!freeSegments.locate(best->position))
fb_assert(false);
freeSegments.fastRemove();
}
return position;
}
//
// TempSpace::releaseSpace
//
2008-12-05 02:20:14 +01:00
// Return previously allocated segment back into not used segments list and
// join it with adjacent segments if found
//
2014-07-17 20:48:46 +02:00
void TempSpace::releaseSpace(offset_t position, FB_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
if (freeSegments.locate(Firebird::locEqual, end))
{
// The next segment is found to be adjacent
Segment* const next_seg = &freeSegments.current();
next_seg->position -= size;
next_seg->size += size;
if (freeSegments.getPrev())
{
// Check the prior segment for being adjacent
Segment* const prior_seg = &freeSegments.current();
if (position == prior_seg->position + prior_seg->size)
{
next_seg->position -= prior_seg->size;
next_seg->size += prior_seg->size;
freeSegments.fastRemove();
}
}
return;
}
2012-03-17 03:26:59 +01:00
if (freeSegments.locate(Firebird::locLess, position))
{
// Check the prior segment for being adjacent
Segment* const prior_seg = &freeSegments.current();
if (position == prior_seg->position + prior_seg->size)
{
prior_seg->size += size;
return;
}
}
freeSegments.add(Segment(position, size));
}
//
// TempSpace::inMemory
//
// Return contiguous chunk of memory if present at given location
//
UCHAR* TempSpace::inMemory(offset_t begin, size_t size) const
{
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;
}
//
// TempSpace::findMemory
//
2008-12-05 02:20:14 +01:00
// Return contiguous chunk of memory and adjust starting offset
// of search range if found
//
UCHAR* TempSpace::findMemory(offset_t& begin, offset_t end, size_t size) const
{
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
while (block && (begin + size <= end))
{
UCHAR* const mem = block->inMemory(local_offset, size);
2008-12-05 02:20:14 +01:00
if (mem)
{
return mem;
}
2008-01-16 11:25:04 +01:00
begin += block->size - local_offset;
local_offset = 0;
block = block->next;
}
2008-01-16 11:25:04 +01: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;
FreeSegmentTree::ConstAccessor accessor(&freeSegments);
for (bool found = accessor.getFirst(); found; found = accessor.getNext())
{
const offset_t size = accessor.current().size;
fb_assert(size != 0);
free += size;
}
offset_t disk = 0;
2014-07-17 20:48:46 +02:00
for (FB_SIZE_T i = 0; i < tempFiles.getCount(); i++)
disk += tempFiles[i]->getSize();
return ((initialBuffer.getCount() + localCacheUsage + disk) == physicalSize);
}
//
// TempSpace::allocateBatch
//
2008-12-05 02:20:14 +01:00
// Allocate up to 'count' contiguous chunks of memory available in free
// 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
// is in memory blocks). Algorithm is very simple and can be improved in future
2008-12-05 02:20:14 +01:00
//
2014-07-17 20:48:46 +02:00
ULONG TempSpace::allocateBatch(ULONG count, FB_SIZE_T minSize, FB_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.
offset_t freeMem = 0;
for (bool found = freeSegments.getFirst(); found; found = freeSegments.getNext())
freeMem += freeSegments.current().size;
freeMem = MIN(freeMem / count, maxSize);
freeMem = MAX(freeMem, minSize);
freeMem = MIN(freeMem, minBlockSize);
freeMem &= ~(FB_ALIGNMENT - 1);
2008-12-05 02:20:14 +01:00
bool is_positioned = freeSegments.getFirst();
while (segments.getCount() < count && is_positioned)
{
Segment* freeSpace = &freeSegments.current();
offset_t freeSeek = freeSpace->position;
const offset_t freeEnd = freeSpace->position + freeSpace->size;
UCHAR* const mem = findMemory(freeSeek, freeEnd, freeMem);
if (mem)
{
fb_assert(freeSeek + freeMem <= freeEnd);
#ifdef DEV_BUILD
offset_t seek1 = freeSeek;
UCHAR* const p = findMemory(seek1, freeEnd, freeMem);
fb_assert(p == mem);
fb_assert(seek1 == freeSeek);
#endif
if (freeSeek != freeSpace->position)
{
const offset_t skip_size = freeSeek - freeSpace->position;
const Segment skip_space(freeSpace->position, skip_size);
freeSpace->position += skip_size;
freeSpace->size -= skip_size;
fb_assert(freeSpace->size != 0);
if (!freeSegments.add(skip_space))
fb_assert(false);
if (!freeSegments.locate(skip_space.position + skip_size))
fb_assert(false);
freeSpace = &freeSegments.current();
}
SegmentInMemory seg;
seg.memory = mem;
seg.position = freeSeek;
seg.size = freeMem;
segments.add(seg);
freeSpace->position += freeMem;
freeSpace->size -= freeMem;
if (!freeSpace->size)
{
is_positioned = freeSegments.fastRemove();
}
}
else
{
is_positioned = freeSegments.getNext();
}
}
return segments.getCount();
}