Line data Source code
1 : /* src/interfaces/ecpg/preproc/util.c */
2 :
3 : #include "postgres_fe.h"
4 :
5 : #include <unistd.h>
6 :
7 : #include "preproc_extern.h"
8 :
9 : static void vmmerror(int error_code, enum errortype type, const char *error, va_list ap) pg_attribute_printf(3, 0);
10 :
11 :
12 : /*
13 : * Handle preprocessor errors and warnings
14 : */
15 : static void
16 0 : vmmerror(int error_code, enum errortype type, const char *error, va_list ap)
17 : {
18 : /* localize the error message string */
19 0 : error = _(error);
20 :
21 0 : fprintf(stderr, "%s:%d: ", input_filename, base_yylineno);
22 :
23 0 : switch (type)
24 : {
25 0 : case ET_WARNING:
26 0 : fprintf(stderr, _("WARNING: "));
27 0 : break;
28 0 : case ET_ERROR:
29 0 : fprintf(stderr, _("ERROR: "));
30 0 : break;
31 : }
32 :
33 0 : vfprintf(stderr, error, ap);
34 :
35 0 : fprintf(stderr, "\n");
36 :
37 : /* If appropriate, set error code to be inspected by ecpg.c */
38 0 : switch (type)
39 : {
40 0 : case ET_WARNING:
41 0 : break;
42 0 : case ET_ERROR:
43 0 : ret_value = error_code;
44 0 : break;
45 : }
46 0 : }
47 :
48 : /* Report an error or warning */
49 : void
50 0 : mmerror(int error_code, enum errortype type, const char *error,...)
51 : {
52 : va_list ap;
53 :
54 0 : va_start(ap, error);
55 0 : vmmerror(error_code, type, error, ap);
56 0 : va_end(ap);
57 0 : }
58 :
59 : /* Report an error and abandon execution */
60 : void
61 0 : mmfatal(int error_code, const char *error,...)
62 : {
63 : va_list ap;
64 :
65 0 : va_start(ap, error);
66 0 : vmmerror(error_code, ET_ERROR, error, ap);
67 0 : va_end(ap);
68 :
69 0 : if (base_yyin)
70 0 : fclose(base_yyin);
71 0 : if (base_yyout)
72 0 : fclose(base_yyout);
73 :
74 0 : if (strcmp(output_filename, "-") != 0 && unlink(output_filename) != 0)
75 0 : fprintf(stderr, _("could not remove output file \"%s\"\n"), output_filename);
76 0 : exit(error_code);
77 : }
78 :
79 : /*
80 : * Basic memory management support
81 : */
82 :
83 : /* malloc + error check */
84 : void *
85 57082 : mm_alloc(size_t size)
86 : {
87 57082 : void *ptr = malloc(size);
88 :
89 57082 : if (ptr == NULL)
90 0 : mmfatal(OUT_OF_MEMORY, "out of memory");
91 :
92 57082 : return ptr;
93 : }
94 :
95 : /* strdup + error check */
96 : char *
97 12350 : mm_strdup(const char *string)
98 : {
99 12350 : char *new = strdup(string);
100 :
101 12350 : if (new == NULL)
102 0 : mmfatal(OUT_OF_MEMORY, "out of memory");
103 :
104 12350 : return new;
105 : }
106 :
107 :
108 : /*
109 : * "Local" memory management support
110 : *
111 : * These functions manage memory that is only needed for a short time
112 : * (processing of one input statement) within the ecpg grammar.
113 : * Data allocated with these is not meant to be freed separately;
114 : * rather it's freed by calling reclaim_local_storage() at the end
115 : * of each statement cycle.
116 : */
117 :
118 : typedef struct loc_chunk
119 : {
120 : struct loc_chunk *next; /* list link */
121 : unsigned int chunk_used; /* index of first unused byte in data[] */
122 : unsigned int chunk_avail; /* # bytes still available in data[] */
123 : char data[FLEXIBLE_ARRAY_MEMBER]; /* actual storage */
124 : } loc_chunk;
125 :
126 : #define LOC_CHUNK_OVERHEAD MAXALIGN(offsetof(loc_chunk, data))
127 : #define LOC_CHUNK_MIN_SIZE 8192
128 :
129 : /* Head of list of loc_chunks */
130 : static loc_chunk *loc_chunks = NULL;
131 :
132 : /*
133 : * Allocate local space of the requested size.
134 : *
135 : * Exits on OOM.
136 : */
137 : void *
138 177174 : loc_alloc(size_t size)
139 : {
140 : void *result;
141 177174 : loc_chunk *cur_chunk = loc_chunks;
142 :
143 : /* Ensure all allocations are adequately aligned */
144 177174 : size = MAXALIGN(size);
145 :
146 : /* Need a new chunk? */
147 177174 : if (cur_chunk == NULL || size > cur_chunk->chunk_avail)
148 : {
149 47900 : size_t chunk_size = Max(size, LOC_CHUNK_MIN_SIZE);
150 :
151 47900 : cur_chunk = mm_alloc(chunk_size + LOC_CHUNK_OVERHEAD);
152 : /* Depending on alignment rules, we could waste a bit here */
153 47900 : cur_chunk->chunk_used = LOC_CHUNK_OVERHEAD - offsetof(loc_chunk, data);
154 47900 : cur_chunk->chunk_avail = chunk_size;
155 : /* New chunk becomes the head of the list */
156 47900 : cur_chunk->next = loc_chunks;
157 47900 : loc_chunks = cur_chunk;
158 : }
159 :
160 177174 : result = cur_chunk->data + cur_chunk->chunk_used;
161 177174 : cur_chunk->chunk_used += size;
162 177174 : cur_chunk->chunk_avail -= size;
163 177174 : return result;
164 : }
165 :
166 : /*
167 : * Copy given string into local storage
168 : */
169 : char *
170 95738 : loc_strdup(const char *string)
171 : {
172 95738 : char *result = loc_alloc(strlen(string) + 1);
173 :
174 95738 : strcpy(result, string);
175 95738 : return result;
176 : }
177 :
178 : /*
179 : * Reclaim local storage when appropriate
180 : */
181 : void
182 47742 : reclaim_local_storage(void)
183 : {
184 : loc_chunk *cur_chunk,
185 : *next_chunk;
186 :
187 95510 : for (cur_chunk = loc_chunks; cur_chunk; cur_chunk = next_chunk)
188 : {
189 47768 : next_chunk = cur_chunk->next;
190 47768 : free(cur_chunk);
191 : }
192 47742 : loc_chunks = NULL;
193 47742 : }
194 :
195 :
196 : /*
197 : * String concatenation support routines. These return "local" (transient)
198 : * storage.
199 : */
200 :
201 : /*
202 : * Concatenate 2 strings, inserting a space between them unless either is empty
203 : */
204 : char *
205 9570 : cat2_str(const char *str1, const char *str2)
206 : {
207 9570 : char *res_str = (char *) loc_alloc(strlen(str1) + strlen(str2) + 2);
208 :
209 9570 : strcpy(res_str, str1);
210 9570 : if (strlen(str1) != 0 && strlen(str2) != 0)
211 6158 : strcat(res_str, " ");
212 9570 : strcat(res_str, str2);
213 9570 : return res_str;
214 : }
215 :
216 : /*
217 : * Concatenate N strings, inserting spaces between them unless they are empty
218 : */
219 : char *
220 2844 : cat_str(int count,...)
221 : {
222 : va_list args;
223 : int i;
224 : char *res_str;
225 :
226 2844 : va_start(args, count);
227 :
228 2844 : res_str = va_arg(args, char *);
229 :
230 : /* now add all other strings */
231 11784 : for (i = 1; i < count; i++)
232 8940 : res_str = cat2_str(res_str, va_arg(args, char *));
233 :
234 2844 : va_end(args);
235 :
236 2844 : return res_str;
237 : }
238 :
239 : /*
240 : * Concatenate 2 strings, with no space between
241 : */
242 : char *
243 100 : make2_str(const char *str1, const char *str2)
244 : {
245 100 : char *res_str = (char *) loc_alloc(strlen(str1) + strlen(str2) + 1);
246 :
247 100 : strcpy(res_str, str1);
248 100 : strcat(res_str, str2);
249 100 : return res_str;
250 : }
251 :
252 : /*
253 : * Concatenate 3 strings, with no space between
254 : */
255 : char *
256 3674 : make3_str(const char *str1, const char *str2, const char *str3)
257 : {
258 3674 : char *res_str = (char *) loc_alloc(strlen(str1) + strlen(str2) + strlen(str3) + 1);
259 :
260 3674 : strcpy(res_str, str1);
261 3674 : strcat(res_str, str2);
262 3674 : strcat(res_str, str3);
263 3674 : return res_str;
264 : }
|