Annotation of libelwix/src/str.c, revision 1.5
1.1 misho 1: /*************************************************************************
2: * (C) 2013 AITNET ltd - Sofia/Bulgaria - <misho@aitnet.org>
3: * by Michael Pounov <misho@elwix.org>
4: *
5: * $Author: misho $
1.5 ! misho 6: * $Id: str.c,v 1.4.12.1 2014/02/12 15:24:50 misho Exp $
1.1 misho 7: *
8: **************************************************************************
9: The ELWIX and AITNET software is distributed under the following
10: terms:
11:
12: All of the documentation and software included in the ELWIX and AITNET
13: Releases is copyrighted by ELWIX - Sofia/Bulgaria <info@elwix.org>
14:
1.4 misho 15: Copyright 2004 - 2014
1.1 misho 16: by Michael Pounov <misho@elwix.org>. All rights reserved.
17:
18: Redistribution and use in source and binary forms, with or without
19: modification, are permitted provided that the following conditions
20: are met:
21: 1. Redistributions of source code must retain the above copyright
22: notice, this list of conditions and the following disclaimer.
23: 2. Redistributions in binary form must reproduce the above copyright
24: notice, this list of conditions and the following disclaimer in the
25: documentation and/or other materials provided with the distribution.
26: 3. All advertising materials mentioning features or use of this software
27: must display the following acknowledgement:
28: This product includes software developed by Michael Pounov <misho@elwix.org>
29: ELWIX - Embedded LightWeight unIX and its contributors.
30: 4. Neither the name of AITNET nor the names of its contributors
31: may be used to endorse or promote products derived from this software
32: without specific prior written permission.
33:
34: THIS SOFTWARE IS PROVIDED BY AITNET AND CONTRIBUTORS ``AS IS'' AND
35: ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
36: IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
37: ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
38: FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
39: DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
40: OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
41: HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
42: LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
43: OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
44: SUCH DAMAGE.
45: */
46: #include "global.h"
47:
48:
49: /*
50: * str_FreeNullTerm() - Free dynamic allocated null terminated array with strings
51: *
52: * @arr = Pointer to array for free
53: * return: none
54: */
1.3 misho 55: void
1.1 misho 56: str_FreeNullTerm(char *** __restrict arr)
57: {
58: char **a;
59:
60: if (arr && *arr) {
61: a = *arr;
62: while (a && *a)
63: e_free(*a++);
64: e_free(*arr);
65: *arr = NULL;
66: }
67: }
68:
69: /*
70: * str_ArgsNum() Parse and calculate number of arguments
71: *
72: * @csArgs = Input arguments line
73: * @csDelim = Delimiter(s) for separate
74: * return: 0 error format; -1 error:: can`t read; >0 ok, number of items
75: */
1.3 misho 76: int
1.1 misho 77: str_ArgsNum(const char *csArgs, const char *csDelim)
78: {
79: register int res;
80: char *pos;
81:
82: if (!csArgs || !csDelim)
83: return -1;
84:
85: for (res = 1, pos = (char*) csArgs; (pos = strpbrk(pos, csDelim)); res++, pos++);
86:
87: return res;
88: }
89:
90: /*
91: * str_ExecArgs() - Build exec arguments from other array
92: *
93: * @psProg = Program name for execute
94: * @oldarg = Arguments array
95: * return: NULL error; !=NULL Allocated execution array(must be e_free)
96: */
97: char **
98: str_ExecArgs(const char *psProg, const char **oldarg)
99: {
100: char **newarg, **el;
101: register int i, num;
102:
103: if (!psProg || !oldarg)
104: return NULL;
105: else
106: newarg = el = NULL;
107:
108: /* count items arguments */
109: for (num = 0; oldarg[num]; num++);
110:
111: /* create and copy new arguments */
112: newarg = e_calloc(num + 2, sizeof(char*));
113: if (!newarg)
114: return NULL;
115: else
116: el = newarg;
117:
118: *el = e_strdup(psProg);
119: el++;
120:
121: for (i = 0; oldarg[i]; i++, el++)
122: *el = e_strdup(oldarg[i]);
123: *el = NULL;
124:
125: return newarg;
126: }
127:
128: /*
129: * str_CopyEnv() - Copy environment to new environment array;
130: *
131: * @oldenv = Environment array
132: * return: NULL error; !=NULL Allocated new environment array(must be e_free)
133: */
134: char **
135: str_CopyEnv(const char **oldenv)
136: {
137: char **newenv, **el;
138: register int i, num;
139:
140: if (!oldenv)
141: return NULL;
142: else
143: newenv = el = NULL;
144:
145: /* count items environment */
146: for (i = num = 0; oldenv[i]; i++)
147: if (*strchr(oldenv[i], '='))
148: num++;
149:
150: /* create and copy new environment */
151: newenv = e_calloc(num + 1, sizeof(char*));
152: if (!newenv)
153: return NULL;
154: else
155: el = newenv;
156:
157: for (i = 0; oldenv[i]; i++)
158: if (*strchr(oldenv[i], '=')) {
159: *el = e_strdup(oldenv[i]);
160: el++;
161: }
162: *el = NULL;
163:
164: return newenv;
165: }
166:
167: /*
168: * str_Ast() - Function for evaluate string like asterisk variable "{text[:[-]#[:#]]}"
169: *
170: * @csString = Input string
171: * return: NULL error, !=NULL Allocated new string evaluated from input string,
172: * must be ait_freeVar()
173: */
174: ait_val_t *
175: str_Ast(const char *csString)
176: {
177: char *ext, *str, *eb;
178: int e[2] = { 0, 0 };
179: ait_val_t *out = NULL;
180:
181: if (!csString)
182: return NULL;
183:
184: if (!strchr(csString, '{') || !strrchr(csString, '}')) {
185: elwix_SetErr(EINVAL, "Invalid input string format ... must be like "
186: "{text[:[-]#[:#]]}");
187: return NULL;
188: } else if (!(out = ait_allocVar()))
189: return NULL;
190: else {
191: AIT_INIT_VAL2(out, string);
192:
193: str = e_strdup(strchr(csString, '{') + 1);
194: *strrchr(str, '}') = 0;
195: }
196:
197: if ((ext = strchr(str, ':'))) {
198: *ext++ = 0;
199: e[0] = strtol(ext, &eb, 0);
200: if ((ext = strchr(eb, ':')))
201: e[1] = strtol(++ext, NULL, 0);
202:
203: /* make cut prefix */
204: if (e[0] >= 0)
205: ext = str + MIN(e[0], strlen(str));
206: else
207: ext = str + MAX(strlen(str) + e[0], 0);
208: /* make cut suffix */
209: if (e[1] > 0)
210: *(ext + MIN(e[1], strlen(ext))) = 0;
211: } else
212: /* ok, clear show */
213: ext = str;
214:
215: AIT_SET_STR(out, ext);
216: e_free(str);
217:
218: return out;
219: }
220:
221: /*
222: * str_Hex2Dig() - Convert from Hex string to digit array
223: *
224: * @psLine = Text string
225: * return: NULL nothing to do or error;
226: * !=0 Allocated new converted data (must be ait_freeVar())
227: */
228: ait_val_t *
229: str_Hex2Dig(const char *psLine)
230: {
231: register int i, j;
232: char *str, szWork[3] = { 0, 0, 0 };
233: ait_val_t *v, s;
234: u_char *b;
235: int n;
236:
237: if (!psLine || !*psLine)
238: return NULL;
239: else {
240: v = ait_allocVar();
241: if (!v)
242: return NULL;
243:
244: /* normalize input string if not even */
245: n = strlen(psLine);
246: if (n % 2)
247: n++;
248: AIT_SET_STRSIZ(&s, n);
249: for (i = strlen(psLine) - 1, j = n - 1, str = AIT_GET_STR(&s), *str = '0';
250: i > -1; i--, j--)
251: str[j] = psLine[i];
252: }
253:
254: AIT_SET_BUFSIZ(v, 0, n / 2);
255: for (i = 0, b = AIT_GET_BUF(v); i < n && str[i * 2]; i++) {
256: strncpy(szWork, &str[i * 2], 2);
257: b[i] = (u_char) strtol(szWork, NULL, 16);
258: }
259:
260: AIT_FREE_VAL(&s);
261: return v;
262: }
263:
264: /*
265: * str_Dig2Hex() - Convert from digit array to Hex string
266: *
267: * @digz = Digits
268: * return: NULL nothing to do or error;
269: * !=0 Allocated new converted string (must be e_free())
270: */
271: char *
272: str_Dig2Hex(ait_val_t *digz)
273: {
274: register int i;
275: char szWork[3] = { 0, 0, 0 }, *str;
276: u_char *b;
277:
278: if (!digz || AIT_ISEMPTY(digz))
279: return NULL;
280:
281: str = e_malloc(AIT_LEN(digz) * 2 + 1);
282: if (!str)
283: return NULL;
284: else
285: memset(str, 0, AIT_LEN(digz) * 2 + 1);
286:
287: for (i = 0, b = AIT_GET_BUF(digz); i < AIT_LEN(digz); i++) {
288: snprintf(szWork, sizeof szWork, "%02hhX", b[i]);
289: strncat(str, szWork, 2);
290: }
291:
292: return str;
293: }
294:
295: /*
296: * str_LTrim() - Remove left whitespaces from text string
297: *
298: * @psLine = Text string
299: * return: 0 nothing to do; !=0 Removed bytes
300: */
1.3 misho 301: int
1.1 misho 302: str_LTrim(char * __restrict psLine)
303: {
304: int pos = 0;
305: char *s;
306:
307: if (!psLine || !*psLine)
308: return 0;
309:
310: for (s = psLine; isspace((u_char) *s); s++);
311: pos = s - psLine;
312:
313: memmove(psLine, s, (strlen(psLine) - pos) + 1);
314: return pos;
315: }
316:
317: /*
318: * str_RTrim() - Remove right whitespaces from text string
319: *
320: * @psLine = Text string
321: * return: 0 nothing to do; !=0 Removed bytes
322: */
1.3 misho 323: int
1.1 misho 324: str_RTrim(char * __restrict psLine)
325: {
326: char *t, *pos;
327:
328: if (!psLine || !*psLine)
329: return 0;
330:
331: pos = psLine + strlen(psLine);
332: for (t = pos - 1; t > psLine && isspace((u_char) *t); t--);
333: *++t = 0;
334:
335: return pos - t;
336: }
337:
338: /*
339: * str_Trim() - Remove left and right whitespaces from text string
340: *
341: * @psLine = Text string
342: * return: 0 nothing to do; !=0 Removed bytes
343: */
1.3 misho 344: int
1.1 misho 345: str_Trim(char * __restrict psLine)
346: {
347: int ret = 0;
348:
349: ret = str_LTrim(psLine);
350: ret += str_RTrim(psLine);
351:
352: return ret;
353: }
354:
355: /*
356: * str_Unquot() - Remove quots from input text string
357: *
358: * @psLine = Text string
359: * return: 0 nothing to do; 1 successful unquoted string
360: */
1.3 misho 361: int
1.1 misho 362: str_Unquot(char * __restrict psLine)
363: {
364: char *pos, *str = NULL;
365: int flg;
366:
367: if (!psLine || !*psLine)
368: return 0;
369:
370: if (*psLine == '"' || *psLine == '\'') {
371: str = e_strdup(psLine + 1);
372: for (pos = str, flg = 0; *pos; flg = ('\\' == *pos), pos++) {
373: if (!flg && *pos == *psLine) {
374: *pos = 0;
375: strlcpy(psLine, str, strlen(psLine) + 1);
376: break;
377: }
378: }
379: e_free(str);
380: return 1;
381: }
382:
383: return 0;
384: }
385:
1.2 misho 386: /*
387: * str_Upper() - Convert all lower characters to upper
388: *
389: * @psLine = Text string
390: * return: 0 nothing to do; !=0 converted chars
391: */
1.3 misho 392: int
1.2 misho 393: str_Upper(char * __restrict psLine)
394: {
395: char *s;
396: register int cx = 0;
397:
398: if (!psLine || !*psLine)
399: return 0;
400:
401: for (s = psLine; *s; s++)
402: if (islower(*s)) {
403: *s = toupper(*s);
404: cx++;
405: }
406:
407: return cx;
408: }
409:
410: /*
411: * str_Lower() - Convert all upper characters to lower
412: *
413: * @psLine = Text string
414: * return: 0 nothing to do; !=0 converted chars
415: */
1.3 misho 416: int
1.2 misho 417: str_Lower(char * __restrict psLine)
418: {
419: char *s;
420: register int cx = 0;
421:
422: if (!psLine || !*psLine)
423: return 0;
424:
425: for (s = psLine; *s; s++)
426: if (isupper(*s)) {
427: *s = tolower(*s);
428: cx++;
429: }
430:
431: return cx;
432: }
1.5 ! misho 433:
! 434: /*
! 435: * str_getString() - Get string from data buffer
! 436: *
! 437: * @data = Data buffer
! 438: * @dlen = Data length
! 439: * @next = Return next position after string if !=NULL
! 440: * return: -1 error or size of string
! 441: */
! 442: int
! 443: str_getString(const u_char * __restrict data, int dlen, char ** __restrict next)
! 444: {
! 445: const u_char *pos;
! 446:
! 447: if (!data || !dlen)
! 448: return -1;
! 449:
! 450: for (pos = data; pos < data + dlen; pos++)
! 451: if (!*pos)
! 452: break;
! 453: if (*pos) {
! 454: elwix_SetErr(ENOEXEC, "Not found null-terminated string");
! 455: return -1;
! 456: }
! 457:
! 458: if (next)
! 459: *next = (char*) pos + 1;
! 460: return pos - data + 1;
! 461: }
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>