xref: /titanic_52/usr/src/lib/libdwarf/common/pro_alloc.c (revision 7fd791373689a6af05e27efec3b1ab556e02aa23)
1*7fd79137SRobert Mustacchi /*
2*7fd79137SRobert Mustacchi 
3*7fd79137SRobert Mustacchi   Copyright (C) 2000,2004 Silicon Graphics, Inc.  All Rights Reserved.
4*7fd79137SRobert Mustacchi   Portions Copyright 2002-2010 Sun Microsystems, Inc. All rights reserved.
5*7fd79137SRobert Mustacchi 
6*7fd79137SRobert Mustacchi   This program is free software; you can redistribute it and/or modify it
7*7fd79137SRobert Mustacchi   under the terms of version 2.1 of the GNU Lesser General Public License
8*7fd79137SRobert Mustacchi   as published by the Free Software Foundation.
9*7fd79137SRobert Mustacchi 
10*7fd79137SRobert Mustacchi   This program is distributed in the hope that it would be useful, but
11*7fd79137SRobert Mustacchi   WITHOUT ANY WARRANTY; without even the implied warranty of
12*7fd79137SRobert Mustacchi   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
13*7fd79137SRobert Mustacchi 
14*7fd79137SRobert Mustacchi   Further, this software is distributed without any warranty that it is
15*7fd79137SRobert Mustacchi   free of the rightful claim of any third person regarding infringement
16*7fd79137SRobert Mustacchi   or the like.  Any license provided herein, whether implied or
17*7fd79137SRobert Mustacchi   otherwise, applies only to this software file.  Patent licenses, if
18*7fd79137SRobert Mustacchi   any, provided herein do not apply to combinations of this program with
19*7fd79137SRobert Mustacchi   other software, or any other product whatsoever.
20*7fd79137SRobert Mustacchi 
21*7fd79137SRobert Mustacchi   You should have received a copy of the GNU Lesser General Public
22*7fd79137SRobert Mustacchi   License along with this program; if not, write the Free Software
23*7fd79137SRobert Mustacchi   Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston MA 02110-1301,
24*7fd79137SRobert Mustacchi   USA.
25*7fd79137SRobert Mustacchi 
26*7fd79137SRobert Mustacchi   Contact information:  Silicon Graphics, Inc., 1500 Crittenden Lane,
27*7fd79137SRobert Mustacchi   Mountain View, CA 94043, or:
28*7fd79137SRobert Mustacchi 
29*7fd79137SRobert Mustacchi   http://www.sgi.com
30*7fd79137SRobert Mustacchi 
31*7fd79137SRobert Mustacchi   For further information regarding this notice, see:
32*7fd79137SRobert Mustacchi 
33*7fd79137SRobert Mustacchi   http://oss.sgi.com/projects/GenInfo/NoticeExplan
34*7fd79137SRobert Mustacchi 
35*7fd79137SRobert Mustacchi */
36*7fd79137SRobert Mustacchi 
37*7fd79137SRobert Mustacchi 
38*7fd79137SRobert Mustacchi 
39*7fd79137SRobert Mustacchi #include "config.h"
40*7fd79137SRobert Mustacchi #include "pro_incl.h"
41*7fd79137SRobert Mustacchi #ifdef HAVE_STDLIB_H
42*7fd79137SRobert Mustacchi #include <stdlib.h>
43*7fd79137SRobert Mustacchi #endif /* HAVE_STDLIB_H */
44*7fd79137SRobert Mustacchi #ifdef HAVE_STRING_H
45*7fd79137SRobert Mustacchi #include <string.h>
46*7fd79137SRobert Mustacchi #endif /* HAVE_STRING_H */
47*7fd79137SRobert Mustacchi #include <malloc.h>
48*7fd79137SRobert Mustacchi 
49*7fd79137SRobert Mustacchi /*
50*7fd79137SRobert Mustacchi  When each block is allocated, there is a two-word structure
51*7fd79137SRobert Mustacchi  allocated at the beginning so the block can go on a list.
52*7fd79137SRobert Mustacchi  The address returned is the address *after* the two pointers
53*7fd79137SRobert Mustacchi  at the start.  But this allows us to be given a pointer to
54*7fd79137SRobert Mustacchi  a generic block, and go backwards to find the list-node.  Then
55*7fd79137SRobert Mustacchi  we can remove this block from it's list without the need to search
56*7fd79137SRobert Mustacchi  through a linked list in order to remove the node.  It also allows
57*7fd79137SRobert Mustacchi  us to 'delete' a memory block without needing the dbg structure.
58*7fd79137SRobert Mustacchi  We still need the dbg structure on allocation so that we know which
59*7fd79137SRobert Mustacchi  linked list to add the block to.
60*7fd79137SRobert Mustacchi 
61*7fd79137SRobert Mustacchi  Only the allocation of the dbg structure itself cannot use _dwarf_p_get_alloc.
62*7fd79137SRobert Mustacchi  That structure should be set up by hand, and the two list pointers
63*7fd79137SRobert Mustacchi  should be initialized to point at the node itself.  That initializes
64*7fd79137SRobert Mustacchi  the doubly linked list.
65*7fd79137SRobert Mustacchi */
66*7fd79137SRobert Mustacchi 
67*7fd79137SRobert Mustacchi #define LIST_TO_BLOCK(lst) ((void*) (((char *)lst) + sizeof(memory_list_t)))
68*7fd79137SRobert Mustacchi #define BLOCK_TO_LIST(blk) ((memory_list_t*) (((char*)blk) - sizeof(memory_list_t)))
69*7fd79137SRobert Mustacchi 
70*7fd79137SRobert Mustacchi 
71*7fd79137SRobert Mustacchi /*
72*7fd79137SRobert Mustacchi   dbg should be NULL only when allocating dbg itself.  In that
73*7fd79137SRobert Mustacchi   case we initialize it to an empty circular doubly-linked list.
74*7fd79137SRobert Mustacchi */
75*7fd79137SRobert Mustacchi 
76*7fd79137SRobert Mustacchi Dwarf_Ptr
77*7fd79137SRobert Mustacchi _dwarf_p_get_alloc(Dwarf_P_Debug dbg, Dwarf_Unsigned size)
78*7fd79137SRobert Mustacchi {
79*7fd79137SRobert Mustacchi     void *sp;
80*7fd79137SRobert Mustacchi     memory_list_t *lp = NULL;
81*7fd79137SRobert Mustacchi     memory_list_t *dbglp = NULL;
82*7fd79137SRobert Mustacchi     memory_list_t *nextblock = NULL;
83*7fd79137SRobert Mustacchi 
84*7fd79137SRobert Mustacchi     /* alloc control struct and data block together for performance reasons */
85*7fd79137SRobert Mustacchi     lp = (memory_list_t *) malloc(size + sizeof(memory_list_t));
86*7fd79137SRobert Mustacchi     if (lp == NULL) {
87*7fd79137SRobert Mustacchi         /* should throw an error */
88*7fd79137SRobert Mustacchi         return NULL;
89*7fd79137SRobert Mustacchi     }
90*7fd79137SRobert Mustacchi 
91*7fd79137SRobert Mustacchi     /* point to 'size' bytes just beyond lp struct */
92*7fd79137SRobert Mustacchi     sp = LIST_TO_BLOCK(lp);
93*7fd79137SRobert Mustacchi     memset(sp, 0, size);
94*7fd79137SRobert Mustacchi 
95*7fd79137SRobert Mustacchi     if (dbg == NULL) {
96*7fd79137SRobert Mustacchi         lp->next = lp->prev = lp;
97*7fd79137SRobert Mustacchi     } else {
98*7fd79137SRobert Mustacchi         /* I always have to draw a picture to understand this part. */
99*7fd79137SRobert Mustacchi 
100*7fd79137SRobert Mustacchi         dbglp = BLOCK_TO_LIST(dbg);
101*7fd79137SRobert Mustacchi         nextblock = dbglp->next;
102*7fd79137SRobert Mustacchi 
103*7fd79137SRobert Mustacchi         /* Insert between dbglp and nextblock */
104*7fd79137SRobert Mustacchi         dbglp->next = lp;
105*7fd79137SRobert Mustacchi         lp->prev = dbglp;
106*7fd79137SRobert Mustacchi         lp->next = nextblock;
107*7fd79137SRobert Mustacchi         nextblock->prev = lp;
108*7fd79137SRobert Mustacchi     }
109*7fd79137SRobert Mustacchi 
110*7fd79137SRobert Mustacchi     return sp;
111*7fd79137SRobert Mustacchi }
112*7fd79137SRobert Mustacchi 
113*7fd79137SRobert Mustacchi /*
114*7fd79137SRobert Mustacchi   This routine is only here in case a caller of an older version of the
115*7fd79137SRobert Mustacchi   library is calling this for some reason.
116*7fd79137SRobert Mustacchi   We will clean up any stray blocks when the session is closed.
117*7fd79137SRobert Mustacchi   No need to remove this block.  In theory the user might be
118*7fd79137SRobert Mustacchi   depending on the fact that we used to just 'free' this.
119*7fd79137SRobert Mustacchi   In theory they might also be
120*7fd79137SRobert Mustacchi   passing a block that they got from libdwarf.  So we don't know if we
121*7fd79137SRobert Mustacchi   should try to remove this block from our global list.  Safest just to
122*7fd79137SRobert Mustacchi   do nothing at this point.
123*7fd79137SRobert Mustacchi 
124*7fd79137SRobert Mustacchi   !!!
125*7fd79137SRobert Mustacchi   This function is deprecated!  Don't call it inside libdwarf or outside of it.
126*7fd79137SRobert Mustacchi   !!!
127*7fd79137SRobert Mustacchi */
128*7fd79137SRobert Mustacchi 
129*7fd79137SRobert Mustacchi void
130*7fd79137SRobert Mustacchi dwarf_p_dealloc(Dwarf_Small * ptr)
131*7fd79137SRobert Mustacchi {
132*7fd79137SRobert Mustacchi     return;
133*7fd79137SRobert Mustacchi }
134*7fd79137SRobert Mustacchi 
135*7fd79137SRobert Mustacchi /*
136*7fd79137SRobert Mustacchi   The dbg structure is not needed here anymore.
137*7fd79137SRobert Mustacchi */
138*7fd79137SRobert Mustacchi 
139*7fd79137SRobert Mustacchi void
140*7fd79137SRobert Mustacchi _dwarf_p_dealloc(Dwarf_P_Debug dbg, Dwarf_Small * ptr) /* ARGSUSED */
141*7fd79137SRobert Mustacchi {
142*7fd79137SRobert Mustacchi   memory_list_t *lp;
143*7fd79137SRobert Mustacchi   lp = BLOCK_TO_LIST(ptr);
144*7fd79137SRobert Mustacchi 
145*7fd79137SRobert Mustacchi   /*
146*7fd79137SRobert Mustacchi     Remove from a doubly linked, circular list.
147*7fd79137SRobert Mustacchi     Read carefully, use a white board if necessary.
148*7fd79137SRobert Mustacchi     If this is an empty list, the following statements are no-ops, and
149*7fd79137SRobert Mustacchi     will write to the same memory location they read from.
150*7fd79137SRobert Mustacchi     This should only happen when we deallocate the dbg structure itself.
151*7fd79137SRobert Mustacchi   */
152*7fd79137SRobert Mustacchi 
153*7fd79137SRobert Mustacchi   lp->prev->next = lp->next;
154*7fd79137SRobert Mustacchi   lp->next->prev = lp->prev;
155*7fd79137SRobert Mustacchi 
156*7fd79137SRobert Mustacchi   free((void*)lp);
157*7fd79137SRobert Mustacchi }
158*7fd79137SRobert Mustacchi 
159*7fd79137SRobert Mustacchi 
160*7fd79137SRobert Mustacchi /*
161*7fd79137SRobert Mustacchi   This routine deallocates all the nodes on the dbg list,
162*7fd79137SRobert Mustacchi   and then deallocates the dbg structure itself.
163*7fd79137SRobert Mustacchi */
164*7fd79137SRobert Mustacchi 
165*7fd79137SRobert Mustacchi void
166*7fd79137SRobert Mustacchi _dwarf_p_dealloc_all(Dwarf_P_Debug dbg)
167*7fd79137SRobert Mustacchi {
168*7fd79137SRobert Mustacchi     memory_list_t *dbglp;
169*7fd79137SRobert Mustacchi 
170*7fd79137SRobert Mustacchi     if (dbg == NULL) {
171*7fd79137SRobert Mustacchi         /* should throw an error */
172*7fd79137SRobert Mustacchi         return;
173*7fd79137SRobert Mustacchi     }
174*7fd79137SRobert Mustacchi 
175*7fd79137SRobert Mustacchi     dbglp = BLOCK_TO_LIST(dbg);
176*7fd79137SRobert Mustacchi     while (dbglp->next != dbglp) {
177*7fd79137SRobert Mustacchi         _dwarf_p_dealloc(dbg, LIST_TO_BLOCK(dbglp->next));
178*7fd79137SRobert Mustacchi     }
179*7fd79137SRobert Mustacchi     if (dbglp->next != dbglp ||
180*7fd79137SRobert Mustacchi         dbglp->prev != dbglp) {
181*7fd79137SRobert Mustacchi 
182*7fd79137SRobert Mustacchi         /* should throw error */
183*7fd79137SRobert Mustacchi         /* For some reason we couldn't free all the blocks? */
184*7fd79137SRobert Mustacchi         return;
185*7fd79137SRobert Mustacchi     }
186*7fd79137SRobert Mustacchi     _dwarf_p_dealloc(NULL, (void*)dbg);
187*7fd79137SRobert Mustacchi }
188*7fd79137SRobert Mustacchi 
189