2002-10-05 21:13:32 +04:00
|
|
|
///-*-C++-*-//////////////////////////////////////////////////////////////////
|
|
|
|
//
|
|
|
|
// The Hoard Multiprocessor Memory Allocator
|
|
|
|
// Contact author: Emery Berger, http://www.cs.utexas.edu/users/emery
|
|
|
|
//
|
|
|
|
// Copyright (c) 1998-2000, The University of Texas at Austin.
|
|
|
|
//
|
|
|
|
// This library is free software; you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU Library General Public License as
|
|
|
|
// published by the Free Software Foundation, http://www.fsf.org.
|
|
|
|
//
|
|
|
|
// 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
|
|
|
|
// Library General Public License for more details.
|
|
|
|
//
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
/*
|
|
|
|
superblock.cpp
|
|
|
|
------------------------------------------------------------------------
|
|
|
|
The superblock class controls a number of blocks (which are
|
|
|
|
allocatable units of memory).
|
|
|
|
------------------------------------------------------------------------
|
|
|
|
Emery Berger | <http://www.cs.utexas.edu/users/emery>
|
|
|
|
Department of Computer Sciences | <http://www.cs.utexas.edu>
|
|
|
|
University of Texas at Austin | <http://www.utexas.edu>
|
|
|
|
========================================================================
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "arch-specific.h"
|
|
|
|
#include "config.h"
|
|
|
|
#include "heap.h"
|
|
|
|
#include "processheap.h"
|
|
|
|
#include "superblock.h"
|
|
|
|
|
2005-03-23 01:59:41 +03:00
|
|
|
using namespace BPrivate;
|
2002-10-05 21:13:32 +04:00
|
|
|
|
2005-03-23 01:59:41 +03:00
|
|
|
|
|
|
|
superblock::superblock(int numBlocks, // The number of blocks in the sb.
|
|
|
|
int szclass, // The size class of the blocks.
|
|
|
|
hoardHeap * o) // The heap that "owns" this sb.
|
|
|
|
:
|
2002-10-05 21:13:32 +04:00
|
|
|
#if HEAP_DEBUG
|
2005-03-23 01:59:41 +03:00
|
|
|
_magic(SUPERBLOCK_MAGIC),
|
2002-10-05 21:13:32 +04:00
|
|
|
#endif
|
2005-03-23 01:59:41 +03:00
|
|
|
_sizeClass(szclass),
|
|
|
|
_numBlocks(numBlocks),
|
|
|
|
_numAvailable(0),
|
|
|
|
_fullness(0), _freeList(NULL), _owner(o), _next(NULL), _prev(NULL)
|
2002-10-05 21:13:32 +04:00
|
|
|
{
|
2005-03-23 01:59:41 +03:00
|
|
|
assert(_numBlocks >= 1);
|
|
|
|
|
|
|
|
// Determine the size of each block.
|
|
|
|
const int blksize = hoardHeap::align(sizeof(block)
|
|
|
|
+ hoardHeap::sizeFromClass(_sizeClass));
|
|
|
|
|
|
|
|
// Make sure this size is in fact aligned.
|
|
|
|
assert((blksize & hoardHeap::ALIGNMENT_MASK) == 0);
|
|
|
|
|
|
|
|
// Set the first block to just past this superblock header.
|
|
|
|
block *b = (block *) hoardHeap::align((unsigned long)(this + 1));
|
|
|
|
|
|
|
|
// Initialize all the blocks,
|
|
|
|
// and insert the block pointers into the linked list.
|
|
|
|
for (int i = 0; i < _numBlocks; i++) {
|
|
|
|
// Make sure the block is on a double-word boundary.
|
|
|
|
assert(((unsigned int)b & hoardHeap::ALIGNMENT_MASK) == 0);
|
|
|
|
new(b) block(this);
|
|
|
|
assert(b->getSuperblock() == this);
|
|
|
|
b->setNext(_freeList);
|
|
|
|
_freeList = b;
|
|
|
|
b = (block *)((char *)b + blksize);
|
|
|
|
}
|
|
|
|
|
|
|
|
_numAvailable = _numBlocks;
|
|
|
|
computeFullness();
|
|
|
|
assert((unsigned long)b <= hoardHeap::align(sizeof(superblock) + blksize * _numBlocks)
|
|
|
|
+ (unsigned long)this);
|
|
|
|
|
|
|
|
hoardLockInit(_upLock, "hoard superblock");
|
2002-10-05 21:13:32 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-03-23 01:59:41 +03:00
|
|
|
superblock *
|
2005-03-23 04:15:33 +03:00
|
|
|
superblock::makeSuperblock(int sizeclass, processHeap *pHeap)
|
2005-03-23 01:59:41 +03:00
|
|
|
{
|
|
|
|
// We need to get more memory.
|
2002-10-05 21:13:32 +04:00
|
|
|
|
2005-03-23 01:59:41 +03:00
|
|
|
char *buf;
|
|
|
|
int numBlocks = hoardHeap::numBlocks(sizeclass);
|
2002-10-05 21:13:32 +04:00
|
|
|
|
2005-03-23 01:59:41 +03:00
|
|
|
// Compute how much memory we need.
|
|
|
|
unsigned long moreMemory;
|
|
|
|
if (numBlocks > 1) {
|
|
|
|
moreMemory = hoardHeap::SUPERBLOCK_SIZE;
|
|
|
|
assert(moreMemory >= hoardHeap::align(sizeof(superblock)
|
|
|
|
+ (hoardHeap::align(sizeof(block)
|
|
|
|
+ hoardHeap::sizeFromClass(sizeclass))) * numBlocks));
|
2002-10-05 21:13:32 +04:00
|
|
|
|
2005-03-23 01:59:41 +03:00
|
|
|
// Get some memory from the process heap.
|
|
|
|
buf = (char *)pHeap->getSuperblockBuffer();
|
|
|
|
} else {
|
|
|
|
// One object.
|
|
|
|
assert(numBlocks == 1);
|
2002-10-05 21:13:32 +04:00
|
|
|
|
2005-03-23 01:59:41 +03:00
|
|
|
size_t blksize = hoardHeap::align(sizeof(block)
|
|
|
|
+ hoardHeap::sizeFromClass(sizeclass));
|
|
|
|
moreMemory = hoardHeap::align(sizeof(superblock) + blksize);
|
2002-10-05 21:13:32 +04:00
|
|
|
|
2005-03-23 01:59:41 +03:00
|
|
|
// Get space from the system.
|
|
|
|
buf = (char *)hoardSbrk(moreMemory);
|
|
|
|
}
|
2002-10-05 21:13:32 +04:00
|
|
|
|
2005-03-23 01:59:41 +03:00
|
|
|
// Make sure that we actually got the memory.
|
|
|
|
if (buf == NULL)
|
|
|
|
return 0;
|
2002-10-05 21:13:32 +04:00
|
|
|
|
2005-03-23 01:59:41 +03:00
|
|
|
buf = (char *)hoardHeap::align((unsigned long)buf);
|
2002-10-05 21:13:32 +04:00
|
|
|
|
2005-03-23 01:59:41 +03:00
|
|
|
// Make sure this buffer is double-word aligned.
|
|
|
|
assert(buf == (char *)hoardHeap::align((unsigned long)buf));
|
|
|
|
assert((((unsigned long)buf) & hoardHeap::ALIGNMENT_MASK) == 0);
|
2002-10-05 21:13:32 +04:00
|
|
|
|
2005-03-23 01:59:41 +03:00
|
|
|
// Instantiate the new superblock in the buffer.
|
|
|
|
return new(buf) superblock(numBlocks, sizeclass, NULL);
|
2002-10-05 21:13:32 +04:00
|
|
|
}
|