8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-29 06:43:03 +01:00
firebird-mirror/src/qli/all.cpp

512 lines
12 KiB
C++

/*
* PROGRAM: JRD Command Oriented Query Language
* MODULE: all.cpp
* DESCRIPTION: Internal block allocator
*
* The contents of this file are subject to the Interbase 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.Inprise.com/IPL.html
*
* Software distributed under the License is distributed on an
* "AS IS" basis, 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 Inprise Corporation
* and its predecessors. Portions created by Inprise Corporation are
* Copyright (C) Inprise Corporation.
*
* All Rights Reserved.
* Contributor(s): ______________________________________.
*/
/***************************************************
THIS MODULE HAS SEVERAL KISSING COUSINS; IF YOU
SHOULD CHANGE ONE OF THE MODULES IN THE FOLLOWING
LIST, PLEASE BE SURE TO CHECK THE OTHERS FOR
SIMILAR CHANGES:
/dsql/all.cpp
/jrd/all.cpp
/pipe/allp.cpp
/qli/all.cpp
/remote/allr.cpp
/gpre/msc.cpp
- THANK YOU
***************************************************/
#include "firebird.h"
#include <string.h>
#include "../qli/dtr.h"
#include "../qli/parse.h"
#include "../qli/compile.h"
#include "../qli/exe.h"
#include "../qli/report.h"
#include "../qli/format.h"
#include "../qli/all_proto.h"
#include "../qli/err_proto.h"
#include "../qli/mov_proto.h"
#include "../jrd/gds_proto.h"
#define BLKDEF(type, root, tail) { sizeof(root), tail },
static const struct
{
SSHORT typ_root_length;
SSHORT typ_tail_length;
} block_sizes[] = {
{0, 0},
#include "../qli/blk.h"
};
#undef BLKDEF
static void extend_pool(PLB, USHORT);
static qli_vec* global_pools;
const int MIN_ALLOCATION = 1024;
BLK ALLQ_alloc( PLB pool, UCHAR type, int count)
{
/**************************************
*
* A L L Q _ a l l o c
*
**************************************
*
* Functional description
* Allocate a block from a given pool and initialize the block.
* This is the primary block allocation routine.
*
**************************************/
if (type <= (SCHAR) type_MIN || type >= (SCHAR) type_MAX)
ERRQ_bugcheck(1); // Msg1 bad block type
// Compute block length
size_t size = block_sizes[type].typ_root_length;
SLONG tail = block_sizes[type].typ_tail_length;
if (tail)
size += count * tail;
size = FB_ALIGN(size, ALIGNMENT);
if (size <= 4 || size > 65535)
ERRQ_bugcheck(2); // Msg2 bad block size
/* Find best fit. Best fit is defined to be the free block of SHORTest
tail. If there isn't a fit, extend the pool and try, try again. */
FRB free;
FRB* best;
size_t best_tail;
while (true)
{
best = NULL;
best_tail = 32767;
for (FRB* ptr = &pool->plb_free; (free = *ptr); ptr = &free->frb_next)
{
if (free->frb_next && (SCHAR *) free >= (SCHAR *) free->frb_next)
ERRQ_bugcheck(434); // memory pool free list is incorrect
else if ((tail = free->frb_header.blk_length - size) >= 0
&& tail < static_cast<SLONG>(best_tail))
{
best = ptr;
best_tail = tail;
if (tail == 0)
break;
}
}
if (best)
break;
extend_pool(pool, size);
}
/* We've got our free block. If there's enough left of the free block
after taking out our block, chop out out block. If not, allocate
the entire free block as our block (a little extra won't hurt). */
free = *best;
FRB block;
if (best_tail > sizeof(frb)) {
USHORT l = free->frb_header.blk_length - size;
block = (FRB) ((SCHAR *) free + l);
free->frb_header.blk_length -= size;
}
else {
*best = free->frb_next;
size += best_tail;
block = free;
}
block->frb_header.blk_type = type;
block->frb_header.blk_pool_id = pool->plb_pool_id;
block->frb_header.blk_length = size;
if (size -= sizeof(blk))
memset((SCHAR *) block + sizeof(blk), 0, size);
return (BLK) block;
}
BLK ALLQ_extend(BLK* pointer, int size)
{
/**************************************
*
* A L L Q _ e x t e n d
*
**************************************
*
* Functional description
* Extend a repeating block, copying the constant part.
*
**************************************/
BLK block = *pointer;
BLK new_blk = (BLK) ALLQ_alloc((PLB) global_pools->vec_object[block->blk_pool_id],
block->blk_type, size);
const int length = MIN(block->blk_length, new_blk->blk_length) - sizeof(blk);
memcpy((SCHAR*) new_blk + sizeof(blk), (SCHAR*) block + sizeof(blk), length);
ALLQ_release((FRB) block);
if (new_blk->blk_type == (SCHAR) type_vec)
((qli_vec*) new_blk)->vec_count = size;
else if (new_blk->blk_type == (SCHAR) type_vcl)
((qli_vcl*) new_blk)->vcl_count = size;
*pointer = new_blk;
return new_blk;
}
void ALLQ_fini()
{
/**************************************
*
* A L L Q _ f i n i
*
**************************************
*
* Functional description
* Get rid of everything.
*
**************************************/
PLB* vector = (PLB*) global_pools->vec_object + global_pools->vec_count;
PLB* until = (PLB*) global_pools->vec_object;
while (--vector >= until)
{
PLB pool = *vector;
if (pool)
{
HNK hunks, hunk;
for (hunks = pool->plb_hunks; hunk = hunks;) {
hunks = hunk->hnk_next;
ALLQ_free(hunk->hnk_address);
}
}
}
}
void ALLQ_free(void* memory)
{
/**************************************
*
* A L L Q _ f r e e
*
**************************************
*
* Functional description
* Give space back to system.
*
**************************************/
gds__free(memory);
}
void ALLQ_init()
{
/**************************************
*
* A L L Q _ i n i t
*
**************************************
*
* Functional description
* Initialize the pool system.
*
**************************************/
qli_vec temp_vector[2];
memcpy(temp_vector, 0, sizeof(temp_vector));
global_pools = temp_vector;
global_pools->vec_count = 1;
global_pools->vec_object[0] = NULL;
PLB pool = ALLQ_pool();
QLI_default_pool = QLI_permanent_pool = pool;
global_pools = (qli_vec*) ALLQ_alloc(pool, type_vec, 10);
global_pools->vec_count = 10;
global_pools->vec_object[0] = (BLK) pool;
}
SCHAR *ALLQ_malloc(SLONG size)
{
/**************************************
*
* A L L Q _ m a l l o c
*
**************************************
*
* Functional description
* Get memory from system.
*
**************************************/
SCHAR *memory = (SCHAR*) gds__alloc(size);
if (memory) {
#ifdef DEBUG_GDS_ALLOC
gds_alloc_flag_unfreed((void *) memory); // Don't care about QLI leaks
#endif
return memory;
}
IBERROR(5); // Msg5 "memory gonzo"
return 0;
}
PLB ALLQ_pool()
{
/**************************************
*
* A L L Q _ p o o l
*
**************************************
*
* Functional description
* Allocate a new pool. This is done by creating a tempory
* pool block on the stack, then allocating a real pool block.
* In SHORT, by mirrors.
*
**************************************/
USHORT pool_id;
// Start by assigning a pool id
for (pool_id = 0; pool_id < global_pools->vec_count; pool_id++)
{
if (!(global_pools->vec_object[pool_id]))
break;
}
if (pool_id >= global_pools->vec_count)
ALLQ_extend((BLK*) &global_pools, pool_id + 10);
plb temp_pool;
global_pools->vec_object[pool_id] = (BLK) &temp_pool;
temp_pool.plb_free = NULL;
temp_pool.plb_hunks = NULL;
temp_pool.plb_pool_id = pool_id;
if (pool_id == 0)
QLI_permanent_pool = &temp_pool;
PLB pool = (PLB) ALLQ_alloc(&temp_pool, type_plb, 0);
pool->plb_pool_id = pool_id;
pool->plb_free = temp_pool.plb_free;
pool->plb_hunks = temp_pool.plb_hunks;
global_pools->vec_object[pool_id] = (BLK) pool;
if (pool_id == 0)
QLI_permanent_pool = pool;
return pool;
}
void ALLQ_push( BLK object, qli_lls** stack)
{
/**************************************
*
* A L L Q _ p u s h
*
**************************************
*
* Functional description
* Push an object on a qli_lls stack.
*
**************************************/
PLB pool = QLI_default_pool;
qli_lls* node = pool->plb_lls;
if (node)
pool->plb_lls = node->lls_next;
else
node = (qli_lls*) ALLQ_alloc(pool, type_lls, 0);
node->lls_object = object;
node->lls_next = *stack;
*stack = node;
}
BLK ALLQ_pop(qli_lls** stack)
{
/**************************************
*
* A L L Q _ p o p
*
**************************************
*
* Functional description
* Pop an object off a linked list stack. Save the node for
* further use.
*
**************************************/
qli_lls* node = *stack;
PLB pool = (PLB) global_pools->vec_object[node->lls_header.blk_pool_id];
*stack = node->lls_next;
node->lls_next = pool->plb_lls;
pool->plb_lls = node;
return node->lls_object;
}
void ALLQ_release( FRB block)
{
/**************************************
*
* A L L Q _ r e l e a s e
*
**************************************
*
* Functional description
* Release a block to its pool. If it is contiguous to
* another free block, combine them. Otherwise link it
* into the free block linked list (kept in ascending order
* of addresses).
*
**************************************/
block->frb_header.blk_type = (SCHAR) type_frb;
UCHAR pool_id = block->frb_header.blk_pool_id;
PLB pool;
if (pool_id >= global_pools->vec_count ||
!(pool = (PLB) global_pools->vec_object[pool_id]))
{
ERRQ_bugcheck(4);
// Msg4 bad pool id
}
FRB prior = NULL;
FRB free;
FRB* ptr;
for (ptr = &pool->plb_free; free = *ptr;
prior = free, ptr = &free->frb_next)
{
if ((SCHAR *) block <= (SCHAR *) free)
break;
}
if ((SCHAR *) block == (SCHAR *) free)
ERRQ_bugcheck(435); // block released twice
// Merge block into list first, then try to combine blocks
block->frb_next = free;
*ptr = block;
// Try to merge the free block with the next one down.
if (free) {
if ((SCHAR *) block + block->frb_header.blk_length == (SCHAR *) free)
{
block->frb_header.blk_length += free->frb_header.blk_length;
block->frb_next = free->frb_next;
}
else if ((SCHAR *) block + block->frb_header.blk_length > (SCHAR *) free)
ERRQ_bugcheck(436); // released block overlaps following free block
}
// Try and merge the block with the prior free block
if (prior) {
if ((SCHAR *) prior + prior->frb_header.blk_length == (SCHAR *) block)
{
prior->frb_header.blk_length += block->frb_header.blk_length;
prior->frb_next = block->frb_next;
}
else if ((SCHAR *) prior + prior->frb_header.blk_length > (SCHAR *) block)
ERRQ_bugcheck(437); // released block overlaps prior free block
}
}
void ALLQ_rlpool( PLB pool)
{
/**************************************
*
* A L L Q _ r l p o o l
*
**************************************
*
* Functional description
* Release a storage pool. This involves nothing more than returning
* hunks to the free hunk list.
*
**************************************/
global_pools->vec_object[pool->plb_pool_id] = NULL;
HNK hunk, hunks;
for (hunks = pool->plb_hunks; hunk = hunks;) {
hunks = hunk->hnk_next;
gds__free(hunk->hnk_address);
}
}
static void extend_pool( PLB pool, USHORT count)
{
/**************************************
*
* e x t e n d _ p o o l
*
**************************************
*
* Functional description
* Extend a pool by at least enough to accomodate a block
* of given size.
*
**************************************/
const SLONG size =
(count + sizeof(hnk) + MIN_ALLOCATION - 1) & ~(MIN_ALLOCATION - 1);
if ((USHORT) size < count)
IBERROR(481); // msg 481 unsuccessful attempt to extend pool beyond 64KB
BLK block = (BLK) ALLQ_malloc(size);
block->blk_length = size;
block->blk_type = (SCHAR) type_frb;
block->blk_pool_id = pool->plb_pool_id;
ALLQ_release((FRB) block);
HNK hunk = (HNK) ALLQ_alloc(pool, type_hnk, 0);
hunk->hnk_address = (SCHAR *) block;
hunk->hnk_length = size;
hunk->hnk_next = pool->plb_hunks;
pool->plb_hunks = hunk;
}