Annotation of libaitio/src/tools.c, revision 1.8.2.1
1.2 misho 1: /*************************************************************************
1.4 misho 2: * (C) 2008 AITNET ltd - Sofia/Bulgaria - <misho@aitnet.org>
3: * by Michael Pounov <misho@elwix.org>
1.2 misho 4: *
5: * $Author: misho $
1.8.2.1 ! misho 6: * $Id: tools.c,v 1.8 2012/02/02 21:32:42 misho Exp $
1.2 misho 7: *
1.4 misho 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:
15: Copyright 2004, 2005, 2006, 2007, 2008, 2009, 2010, 2011
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: */
1.2 misho 46: #include "global.h"
47: #include "aitio.h"
48:
49:
50: /*
51: * io_LTrimStr() Remove left whitespaces from text string
52: * @psLine = Text string
53: * return: 0 nothing to do; !=0 Removed bytes
54: */
1.5 misho 55: inline int
56: io_LTrimStr(u_char * __restrict psLine)
1.2 misho 57: {
58: int pos = 0;
59: u_char *s;
60:
61: if (!psLine || !*psLine)
62: return 0;
63:
64: for (s = psLine; isspace(*s); s++);
65: pos = s - psLine;
66:
67: memmove(psLine, s, (strlen((char*) psLine) - pos) + 1);
68: return pos;
69: }
70:
71: /*
72: * io_RTrimStr() Remove right whitespaces from text string
73: * @psLine = Text string
74: * return: 0 nothing to do; !=0 Removed bytes
75: */
1.5 misho 76: inline int
77: io_RTrimStr(u_char * __restrict psLine)
1.2 misho 78: {
79: u_char *t, *pos;
80:
81: if (!psLine || !*psLine)
82: return 0;
83:
84: pos = psLine + strlen((char*) psLine);
85: for (t = pos - 1; t > psLine && isspace(*t); t--);
86: *++t = 0;
87:
88: return pos - t;
89: }
90:
91: /*
92: * io_TrimStr() Remove left and right whitespaces from text string
93: * @psLine = Text string
94: * return: 0 nothing to do; !=0 Removed bytes
95: */
1.5 misho 96: inline int
97: io_TrimStr(u_char * __restrict psLine)
1.2 misho 98: {
99: int ret = 0;
100:
101: ret = io_LTrimStr(psLine);
102: ret += io_RTrimStr(psLine);
103:
104: return ret;
105: }
106:
107: /*
108: * io_UnquotStr() Remove quots from input text string
109: * @psLine = Text string
110: * return: 0 nothing to do; 1 successful unquoted string
111: */
1.5 misho 112: inline int
113: io_UnquotStr(u_char * __restrict psLine)
1.2 misho 114: {
115: char *pos, *str = NULL;
116: int flg;
117:
118: if (!psLine)
119: return 0;
120:
121: switch (*psLine) {
122: case '`':
123: case '"':
124: case '\'':
125: str = strdup((char*) psLine + 1);
126: for (pos = str, flg = 0; *pos; flg = ('\\' == *pos), pos++) {
127: if (!flg && *pos == *psLine) {
128: *pos = 0;
129: strlcpy((char*) psLine, str, strlen((char*) psLine) + 1);
130: break;
131: }
132: }
133: free(str);
134: return 1;
135: }
136:
137: return 0;
138: }
139:
140: /*
1.3 misho 141: * io_Ch2Hex() Convert from Char string to Hex string
1.2 misho 142: * @psLine = Text string
143: * @lineLen = Length of Text string
1.3 misho 144: * return: NULL nothing to do or error; !=0 Allocated new converted data without term\0 (must be free)
1.2 misho 145: */
1.5 misho 146: inline u_char *
147: io_Ch2Hex(u_char *psLine, int lineLen)
1.2 misho 148: {
149: register int i;
1.3 misho 150: char szWork[3];
151: u_char *str;
1.2 misho 152:
153: if (!psLine || !*psLine || !lineLen)
154: return NULL;
155:
1.3 misho 156: str = malloc(lineLen / 2);
1.2 misho 157: if (!str) {
158: LOGERR;
159: return NULL;
160: } else
1.3 misho 161: memset(str, 0, lineLen / 2);
1.2 misho 162:
1.3 misho 163: for (i = 0; i < lineLen && psLine[i * 2]; i++) {
164: strlcpy(szWork, (char*) &psLine[i * 2], 3);
165: str[i] = (u_char) strtol(szWork, NULL, 16);
1.2 misho 166: }
167:
168: return str;
169: }
170:
171:
172: /*
1.3 misho 173: * io_Hex2Ch() Convert from Hex string to Char string
1.2 misho 174: * @psLine = Text string
175: * @lineLen = Length of Text string
176: * return: NULL nothing to do or error; !=0 Allocated new converted string(must be free)
177: */
1.5 misho 178: inline char *
179: io_Hex2Ch(u_char *psLine, int lineLen)
1.2 misho 180: {
181: register int i;
182: char szWork[3], *str;
183:
184: if (!psLine || !*psLine || !lineLen)
185: return NULL;
186:
1.3 misho 187: str = malloc(lineLen * 2 + 1);
1.2 misho 188: if (!str) {
189: LOGERR;
190: return NULL;
191: } else
1.3 misho 192: memset(str, 0, lineLen * 2 + 1);
1.2 misho 193:
1.3 misho 194: for (i = 0; i <= lineLen; i++) {
195: memset(szWork, 0, 3);
196: snprintf(szWork, 3, "%02X", (u_char) psLine[i]);
197: strncat(str, szWork, 2);
1.2 misho 198: }
199:
200: return str;
201: }
1.5 misho 202:
203: /*
204: * io_CopyEnv() Copy environment to new environment array;
205: * @oldenv = Environment array
206: * return: NULL error; !=NULL Allocated new environment array(must be free)
207: */
208: char **
209: io_CopyEnv(const char **oldenv)
210: {
211: char **newenv, **el;
212: register int i, num;
213:
214: if (!oldenv)
215: return NULL;
216: else
217: newenv = el = NULL;
218:
219: /* count items environment */
220: for (i = num = 0; oldenv[i]; i++)
221: if (*strchr(oldenv[i], '='))
222: num++;
223:
224: /* create and copy new environment */
225: newenv = calloc(num + 1, sizeof(char*));
226: if (!newenv) {
227: LOGERR;
228: return NULL;
229: } else
230: el = newenv;
231:
232: for (i = 0; oldenv[i]; i++)
233: if (*strchr(oldenv[i], '=')) {
234: *el = strdup(oldenv[i]);
235: el++;
236: }
237: *el = NULL;
238:
239: return newenv;
240: }
241:
242: /*
243: * io_ExecArgs() Build exec arguments from other array
244: * @psProg = Program name for execute
245: * @oldarg = Arguments array
246: * return: NULL error; !=NULL Allocated execution array(must be free)
247: */
248: char **
249: io_ExecArgs(const char *psProg, const char **oldarg)
250: {
251: char **newarg, **el;
252: register int i, num;
253:
254: if (!psProg || !oldarg)
255: return NULL;
256: else
257: newarg = el = NULL;
258:
259: /* count items arguments */
260: for (num = 0; oldarg[num]; num++);
261:
262: /* create and copy new arguments */
263: newarg = calloc(num + 2, sizeof(char*));
264: if (!newarg) {
265: LOGERR;
266: return NULL;
267: } else
268: el = newarg;
269:
270: *el = strdup(psProg);
271: el++;
272:
273: for (i = 0; oldarg[i]; i++, el++)
274: *el = strdup(oldarg[i]);
275: *el = NULL;
276:
277: return newarg;
278: }
279:
280: /*
281: * io_FreeNullTerm() Free dynamic allocated null terminated array with strings
282: * @arr = Pointer to array for free
283: * return: none
284: */
285: inline void
286: io_FreeNullTerm(char *** __restrict arr)
287: {
288: char **a;
289:
290: if (arr && *arr) {
291: a = *arr;
292: while (a && *a)
293: free(*a++);
294: free(*arr);
295: *arr = NULL;
296: }
297: }
1.6 misho 298:
299: /*
1.8.2.1 ! misho 300: * io_Path2File() - Parse and make path/filename pair
! 301: *
! 302: * @csArgs = Input argument line
! 303: * @psPath = Output Path, if ==NULL path not returned
! 304: * @pathLen = Size of path array
! 305: * @psFile = Output File
! 306: * @fileLen = Size of file array
! 307: * return: 0 error format; -1 error:: can`t read; >0 ok, number of readed items
! 308: */
! 309: inline int
! 310: io_Path2File(const char * __restrict csArgs, char * __restrict psPath,
! 311: int pathLen, char * __restrict psFile, int fileLen)
! 312: {
! 313: char *pos, *psBuf;
! 314:
! 315: if (!csArgs || !psFile || !fileLen)
! 316: return -1;
! 317: if (psPath && !pathLen)
! 318: return -1;
! 319:
! 320: psBuf = strdup(csArgs);
! 321: if (!psBuf) {
! 322: LOGERR;
! 323: return -1;
! 324: }
! 325:
! 326: pos = strrchr(psBuf, '/');
! 327: if (!pos) {
! 328: strlcpy(psFile, psBuf, fileLen);
! 329:
! 330: free(psBuf);
! 331: return 1;
! 332: } else
! 333: *pos++ = 0;
! 334:
! 335: strlcpy(psFile, pos, fileLen);
! 336: if (psPath)
! 337: strlcpy(psPath, psBuf, pathLen);
! 338:
! 339: free(psBuf);
! 340: return 2;
! 341: }
! 342:
! 343: /*
1.6 misho 344: * io_ether_ntoa() Convert ethernet address to string
345: * @n = ethernet address structure, like struct ether_addr
346: * @a = string
347: * @len = string length
348: * return: NULL error or !=NULL string a
349: */
350: inline char *
351: io_ether_ntoa(const struct io_ether_addr *n, char * __restrict a, int len)
352: {
353: if (!n || !a)
354: return NULL;
355:
356: memset(a, 0, len);
357: if (snprintf(a, len, "%02x:%02x:%02x:%02x:%02x:%02x",
358: n->ether_addr_octet[0], n->ether_addr_octet[1],
359: n->ether_addr_octet[2], n->ether_addr_octet[3],
360: n->ether_addr_octet[4], n->ether_addr_octet[5]) < 17)
361: return NULL;
362:
363: return a;
364: }
365:
366: /*
367: * io_ether_aton() Convert string to ethernet address
368: * @a = string
369: * @e = ethernet address structure, like struct ether_addr
370: * return: NULL error or !=NULL ethernet address structure
371: */
372: inline struct io_ether_addr *
373: io_ether_aton(const char *a, struct io_ether_addr *e)
374: {
375: int i;
376: u_int o0, o1, o2, o3, o4, o5;
377:
378: if (!a || !e)
379: return NULL;
380:
381: i = sscanf(a, "%x:%x:%x:%x:%x:%x", &o0, &o1, &o2, &o3, &o4, &o5);
382: if (i != 6)
383: return NULL;
384:
385: e->ether_addr_octet[0] = o0;
386: e->ether_addr_octet[1] = o1;
387: e->ether_addr_octet[2] = o2;
388: e->ether_addr_octet[3] = o3;
389: e->ether_addr_octet[4] = o4;
390: e->ether_addr_octet[5] = o5;
391:
392: return e;
393: }
1.7 misho 394:
395: /*
396: * io_n2port() Extract port from network structure
397: * @addr = Address
398: * return: 0 not supported family type or port number
399: */
400: inline u_short
401: io_n2port(io_sockaddr_t * __restrict addr)
402: {
403: u_short port = 0;
404:
405: if (!addr)
406: return port;
407:
408: switch (addr->sa.sa_family) {
409: case AF_INET:
410: return ntohs(addr->sin.sin_port);
411: case AF_INET6:
412: return ntohs(addr->sin6.sin6_port);
413: default:
414: io_SetErr(ESOCKTNOSUPPORT, "Unsuported address family %d",
415: addr->sa.sa_family);
416: break;
417: }
418:
419: return port;
420: }
421:
422: /*
423: * io_n2addr() Extract address from network structure
424: * @addr = Address
425: * @val = Value for store string address
426: * return: NULL error or !=NULL string address from val
427: */
428: const char *
429: io_n2addr(io_sockaddr_t * __restrict addr, ait_val_t * __restrict val)
430: {
431: char str[INET6_ADDRSTRLEN] = { 0 };
432: const char *ret = NULL;
433:
434: if (!addr || !val)
435: return ret;
436:
437: switch (addr->sa.sa_family) {
438: case AF_INET:
439: if (!inet_ntop(AF_INET, &addr->sin.sin_addr, str, INET_ADDRSTRLEN)) {
440: LOGERR;
441: return ret;
442: } else
443: ret = str;
444: break;
445: case AF_INET6:
446: if (!inet_ntop(AF_INET6, &addr->sin6.sin6_addr, str, INET6_ADDRSTRLEN)) {
447: LOGERR;
448: return ret;
449: } else
450: ret = str;
451: break;
452: default:
453: io_SetErr(EPROTONOSUPPORT, "Unsuported address family %d",
454: addr->sa.sa_family);
455: return ret;
456: }
457:
1.8 misho 458: if (!AIT_ISEMPTY(val))
459: AIT_FREE_VAL(val);
1.7 misho 460: AIT_SET_STR(val, ret);
1.8 misho 461: return (const char*) AIT_GET_STR(val);
1.7 misho 462: }
463:
464: /*
465: * io_gethostbyname() Get host and port and make network structure
466: * @psHost = Hostname
467: * @port = Port
468: * @addr = Network address structure
469: * return: NULL error or !=NULL network structure
470: */
471: io_sockaddr_t *
472: io_gethostbyname(const char *psHost, u_short port, io_sockaddr_t * __restrict addr)
473: {
474: struct hostent *host;
475:
476: if (!psHost || !addr)
477: return NULL;
478:
479: /* resolver */
480: if (!addr->sa.sa_family)
481: host = gethostbyname(psHost);
482: else
483: host = gethostbyname2(psHost, addr->sa.sa_family);
484: if (!host) {
485: io_SetErr(EINVAL, "Resolver #%d - %s", h_errno, hstrerror(h_errno));
486: return NULL;
487: }
488:
489: memset(addr, 0, sizeof(io_sockaddr_t));
490: switch (host->h_addrtype) {
491: case AF_INET:
492: addr->sin.sin_len = sizeof(struct sockaddr_in);
493: addr->sin.sin_family = AF_INET;
494: addr->sin.sin_port = htons(port);
495: memcpy(&addr->sin.sin_addr, host->h_addr, sizeof addr->sin.sin_addr);
496: return addr;
497: case AF_INET6:
498: addr->sin6.sin6_len = sizeof(struct sockaddr_in6);
499: addr->sin6.sin6_family = AF_INET6;
500: addr->sin6.sin6_port = htons(port);
501: memcpy(&addr->sin6.sin6_addr, host->h_addr, sizeof addr->sin6.sin6_addr);
502: return addr;
503: default:
504: io_SetErr(EPROTONOSUPPORT, "Unsuported address family %d",
505: host->h_addrtype);
506: break;
507: }
508:
509: return NULL;
510: }
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>