/* * 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 #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(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; }