/* * %CopyrightBegin% * * Copyright Ericsson AB 2000-2021. All Rights Reserved. * * Licensed under the Apache License, Version 2.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.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. * * %CopyrightEnd% */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include "sys.h" #include "beam_catches.h" #include "global.h" /* R14B04 has about 380 catches when starting erlang */ #define DEFAULT_TABSIZE (1024) typedef struct { ErtsCodePtr cp; unsigned cdr; } beam_catch_t; #ifdef DEBUG # define IF_DEBUG(x) x #else # define IF_DEBUG(x) #endif struct bc_pool { int free_list; unsigned high_mark; unsigned tabsize; beam_catch_t *beam_catches; /* * Note that the 'beam_catches' area is shared by pools. Used slots * are readonly as long as the module is not purgable. The free-list is * protected by the code_ix lock. */ IF_DEBUG(int is_staging;) }; static struct bc_pool bccix[ERTS_NUM_CODE_IX]; void beam_catches_init(void) { int i; bccix[0].tabsize = DEFAULT_TABSIZE; bccix[0].free_list = -1; bccix[0].high_mark = 0; bccix[0].beam_catches = erts_alloc(ERTS_ALC_T_CATCHES, sizeof(beam_catch_t)*DEFAULT_TABSIZE); IF_DEBUG(bccix[0].is_staging = 0); for (i=1; iis_staging); /* * Allocate from free_list while it is non-empty. * If free_list is empty, allocate at high_mark. */ if (p->free_list >= 0) { i = p->free_list; p->free_list = p->beam_catches[i].cdr; } else { if (p->high_mark >= p->tabsize) { /* No free slots and table is full: realloc table */ beam_catch_t* prev_vec = p->beam_catches; unsigned newsize = p->tabsize*2; p->beam_catches = erts_alloc(ERTS_ALC_T_CATCHES, newsize*sizeof(beam_catch_t)); sys_memcpy(p->beam_catches, prev_vec, p->tabsize*sizeof(beam_catch_t)); gc_old_vec(prev_vec); p->tabsize = newsize; } i = p->high_mark++; } p->beam_catches[i].cp = cp; p->beam_catches[i].cdr = cdr; if (cppp) { *cppp = &p->beam_catches[i].cp; } return i; } ErtsCodePtr beam_catches_car(unsigned i) { struct bc_pool* p = &bccix[erts_active_code_ix()]; if (i >= p->tabsize ) { erts_exit(ERTS_ERROR_EXIT, "beam_catches_delmod: index %#x is out of range\r\n", i); } return p->beam_catches[i].cp; } ErtsCodePtr beam_catches_car_staging(unsigned i) { struct bc_pool* p = &bccix[erts_staging_code_ix()]; if (i >= p->tabsize ) { erts_exit(ERTS_ERROR_EXIT, "beam_catches_delmod: index %#x is out of range\r\n", i); } return p->beam_catches[i].cp; } void beam_catches_delmod(unsigned head, const BeamCodeHeader *hdr, unsigned code_bytes, ErtsCodeIndex code_ix) { struct bc_pool* p = &bccix[code_ix]; const char *code_start; unsigned i, cdr; code_start = (const char*)hdr; ASSERT((code_ix == erts_active_code_ix()) != bccix[erts_staging_code_ix()].is_staging); for(i = head; i != (unsigned)-1;) { const char *catch_addr = (char*)p->beam_catches[i].cp; if (i >= p->tabsize) { erts_exit(ERTS_ERROR_EXIT, "beam_catches_delmod: index %#x is out of range\r\n", i); } if (!ErtsInArea(catch_addr, code_start, code_bytes)) { erts_exit(ERTS_ERROR_EXIT, "beam_catches_delmod: item %#x has cp %p which is not " "in module's range [%p,%p[\r\n", i, p->beam_catches[i].cp, code_start, &code_start[code_bytes]); } p->beam_catches[i].cp = NULL; cdr = p->beam_catches[i].cdr; p->beam_catches[i].cdr = p->free_list; p->free_list = i; i = cdr; } }