@cfg_get.<group_name>.<var_name> is documented
[sip-router] / select_buf.c
1 /*
2  * $Id$
3  *
4  * Copyright (C) 2005-2006 iptelorg GmbH
5  *
6  * This file is part of ser, a free SIP server.
7  *
8  * ser is free software; you can redistribute it and/or modify
9  * it under the terms of the GNU General Public License as published by
10  * the Free Software Foundation; either version 2 of the License, or
11  * (at your option) any later version
12  *
13  * For a license to use the ser software under conditions
14  * other than those described here, or to purchase support for this
15  * software, please contact iptel.org by e-mail at the following addresses:
16  *    info@iptel.org
17  *
18  * ser is distributed in the hope that it will be useful,
19  * but WITHOUT ANY WARRANTY; without even the implied warranty of
20  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
21  * GNU General Public License for more details.
22  *
23  * You should have received a copy of the GNU General Public License
24  * along with this program; if not, write to the Free Software
25  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
26  *
27  * History:
28  * --------
29  *      2006-06-16  static buffer for select results (mma)
30  *                  each process owns a separate space
31  *                  each request starts using the buffer from the start
32  *
33  */
34
35 #include <stdio.h>
36 #include <stdlib.h>
37 #include <string.h>
38 #include <unistd.h>
39 #include <ctype.h>
40
41 #include "dprint.h"
42 #include "mem/mem.h"
43 #include "str.h"
44 #include "ut.h"
45
46 /*
47  * Placeholder for the buffer
48  *
49  * two buffers are actually used to cover the different size requests
50  * assuming that resize can move the result to newly allocated space
51  * and comparing two selects from the script could require two static buffers
52  *
53  * if more static buffers need to be valid at the same time change
54  * the following constant
55  */
56
57 #define MAX_BUFFERS 2
58 #define BUFFER_GRANULARITY 256
59
60 typedef struct stat_buffer_ {
61         char *b;
62         int size;
63         int offset;
64 } stat_buffer_t;
65
66 static stat_buffer_t buffer[MAX_BUFFERS];
67 static int active_buffer=-1;
68
69 #define ALLOC_SIZE(req_size) (((req_size/BUFFER_GRANULARITY)+1)*BUFFER_GRANULARITY)
70
71 static int allocate_buffer(int req_size) {
72         void *b;
73         int size=ALLOC_SIZE(req_size);
74         
75         if (buffer[active_buffer].b == NULL) {
76                 if ((buffer[active_buffer].b=pkg_malloc(size))==NULL)
77                         return 0;
78                 buffer[active_buffer].size=size;
79                 buffer[active_buffer].offset=0;
80                 return 1;
81         }
82         
83         active_buffer = (active_buffer?active_buffer:MAX_BUFFERS)-1;
84         if (buffer[active_buffer].size >= req_size) {
85                 buffer[active_buffer].offset = 0;
86                 return 1;
87         }
88         
89         if ((b=pkg_realloc(buffer[active_buffer].b,size))) {
90                 buffer[active_buffer].b=b;
91                 buffer[active_buffer].size=size;
92                 buffer[active_buffer].offset=0;
93                 return 1;
94         }
95         
96         return 0;
97 }
98
99 /*
100  * Request for space from buffer
101  *
102  * Returns:  NULL  memory allocation failure (no more space)
103  *           pointer to the space on success
104  */
105
106 char* get_static_buffer(int req_size) {
107         char *p = NULL;
108         if ((buffer[active_buffer].size >= buffer[active_buffer].offset + req_size)
109                         || (allocate_buffer(req_size))) {
110                 /* enough space in current buffer or allocation successful */
111                 p = buffer[active_buffer].b+buffer[active_buffer].offset;
112                 buffer[active_buffer].offset += req_size;       
113                 return p;
114         }
115         return NULL;
116 }
117
118 /* Internal function - called before request is going to be processed
119  *
120  * Reset offset to unused space
121  */
122
123 int reset_static_buffer() {
124         int i;
125
126         if (active_buffer == -1) {
127                 memset(buffer, 0, sizeof(buffer));
128         } else {
129                 for (i=0; i<MAX_BUFFERS; i++)
130                         buffer[i].offset=0;
131         }
132         active_buffer=0;
133         return 0;
134 }
135
136 int str_to_static_buffer(str* res, str* s)
137 {
138         res->s = get_static_buffer(s->len);
139         if (!res->s) return -1;
140         memcpy(res->s, s->s, s->len);
141         res->len = s->len;
142         return 0;
143 }
144
145 int int_to_static_buffer(str* res, int val)
146 {
147         char *c;
148         c = int2str(abs(val), &res->len);
149         res->s = get_static_buffer(res->len+(val<0)?1:0);
150         if (!res->s) return -1;
151         if (val < 0) {
152                 res->s[0] = '-';        
153                 memcpy(res->s+1, c, res->len);
154                 res->len++;
155         }
156         else {
157                 memcpy(res->s, c, res->len);
158         }
159         return 0;
160 }
161
162 int uint_to_static_buffer(str* res, unsigned int val)
163 {
164         char *c;
165         c = int2str(val, &res->len);
166         res->s = get_static_buffer(res->len);
167         if (!res->s) return -1;
168         memcpy(res->s, c, res->len);
169         return 0;
170 }
171
172 int uint_to_static_buffer_ex(str* res, unsigned int val, int base, int pad)
173 {
174         char *c;
175         c = int2str_base_0pad(val, &res->len, base, pad); 
176         res->s = get_static_buffer(res->len);
177         if (!res->s) return -1;
178         memcpy(res->s, c, res->len);
179         return 0;
180 }
181