Annotation of libelwix/src/str.c, revision 1.11

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.11    ! misho       6: * $Id: str.c,v 1.10.18.1 2024/10/26 14:40:36 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.11    ! misho      15: Copyright 2004 - 2024
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 };
1.7       misho     233:        ait_val_t *v, s = AIT_VAL_INIT;
1.1       misho     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++) {
1.9       misho     288:                snprintf(szWork, sizeof szWork, "%02hhX", b[i]);
1.10      misho     289:                strcat(str, szWork);
1.9       misho     290:        }
                    291: 
                    292:        return str;
                    293: }
                    294: 
                    295: /*
                    296:  * str_Dig2Hex2() - Convert from digit array to Hex string
                    297:  *
                    298:  * @digz = Digits array
                    299:  * @diglen = Array length
                    300:  * return: NULL nothing to do or error; 
                    301:  *             !=0 Allocated new converted string (must be e_free())
                    302: */
                    303: char *
                    304: str_Dig2Hex2(u_char * __restrict digz, int diglen)
                    305: {
                    306:        register int i;
                    307:        char szWork[3] = { 0, 0, 0 }, *str;
                    308:        u_char *b;
                    309: 
                    310:        if (!digz || !diglen)
                    311:                return NULL;
                    312: 
                    313:        str = e_malloc(diglen * 2 + 1);
                    314:        if (!str)
                    315:                return NULL;
                    316:        else
                    317:                memset(str, 0, diglen * 2 + 1);
                    318: 
                    319:        for (i = 0, b = digz; i < diglen; i++) {
1.1       misho     320:                snprintf(szWork, sizeof szWork, "%02hhX", b[i]);
1.10      misho     321:                strcat(str, szWork);
1.1       misho     322:        }
                    323: 
                    324:        return str;
                    325: }
                    326: 
                    327: /*
                    328:  * str_LTrim() - Remove left whitespaces from text string
                    329:  *
                    330:  * @psLine = Text string
                    331:  * return: 0 nothing to do; !=0 Removed bytes
                    332:  */
1.3       misho     333: int
1.1       misho     334: str_LTrim(char * __restrict psLine)
                    335: {
                    336:        int pos = 0;
                    337:        char *s;
                    338: 
                    339:        if (!psLine || !*psLine)
                    340:                return 0;
                    341: 
                    342:        for (s = psLine; isspace((u_char) *s); s++);
                    343:        pos = s - psLine;
                    344: 
                    345:        memmove(psLine, s, (strlen(psLine) - pos) + 1);
                    346:        return pos;
                    347: }
                    348: 
                    349: /*
                    350:  * str_RTrim() - Remove right whitespaces from text string
                    351:  *
                    352:  * @psLine = Text string
                    353:  * return: 0 nothing to do; !=0 Removed bytes
                    354:  */
1.3       misho     355: int
1.1       misho     356: str_RTrim(char * __restrict psLine)
                    357: {
                    358:        char *t, *pos;
                    359: 
                    360:        if (!psLine || !*psLine)
                    361:                return 0;
                    362: 
                    363:        pos = psLine + strlen(psLine);
                    364:        for (t = pos - 1; t > psLine && isspace((u_char) *t); t--);
                    365:        *++t = 0;
                    366: 
                    367:        return pos - t;
                    368: }
                    369: 
                    370: /*
                    371:  * str_Trim() - Remove left and right whitespaces from text string
                    372:  *
                    373:  * @psLine = Text string
                    374:  * return: 0 nothing to do; !=0 Removed bytes
                    375:  */
1.3       misho     376: int
1.1       misho     377: str_Trim(char * __restrict psLine)
                    378: {
                    379:        int ret = 0;
                    380: 
                    381:        ret = str_LTrim(psLine);
                    382:        ret += str_RTrim(psLine);
                    383: 
                    384:        return ret;
                    385: }
                    386: 
                    387: /*
                    388:  * str_Unquot() - Remove quots from input text string 
                    389:  *
                    390:  * @psLine = Text string
                    391:  * return: 0 nothing to do; 1 successful unquoted string
                    392:  */
1.3       misho     393: int
1.1       misho     394: str_Unquot(char * __restrict psLine)
                    395: {
                    396:        char *pos, *str = NULL;
                    397:        int flg;
                    398: 
                    399:        if (!psLine || !*psLine)
                    400:                return 0;
                    401: 
                    402:        if (*psLine == '"' || *psLine == '\'') {
                    403:                str = e_strdup(psLine + 1);
                    404:                for (pos = str, flg = 0; *pos; flg = ('\\' == *pos), pos++) {
                    405:                        if (!flg && *pos == *psLine) {
                    406:                                *pos = 0;
                    407:                                strlcpy(psLine, str, strlen(psLine) + 1);
                    408:                                break;
                    409:                        }
                    410:                }
                    411:                e_free(str);
                    412:                return 1;
                    413:        }
                    414: 
                    415:        return 0;
                    416: }
                    417: 
1.2       misho     418: /*
                    419:  * str_Upper() - Convert all lower characters to upper
                    420:  *
                    421:  * @psLine = Text string
                    422:  * return: 0 nothing to do; !=0 converted chars
                    423:  */
1.3       misho     424: int
1.2       misho     425: str_Upper(char * __restrict psLine)
                    426: {
                    427:        char *s;
                    428:        register int cx = 0;
                    429: 
                    430:        if (!psLine || !*psLine)
                    431:                return 0;
                    432: 
                    433:        for (s = psLine; *s; s++)
                    434:                if (islower(*s)) {
                    435:                        *s = toupper(*s);
                    436:                        cx++;
                    437:                }
                    438: 
                    439:        return cx;
                    440: }
                    441: 
                    442: /*
                    443:  * str_Lower() - Convert all upper characters to lower
                    444:  *
                    445:  * @psLine = Text string
                    446:  * return: 0 nothing to do; !=0 converted chars
                    447:  */
1.3       misho     448: int
1.2       misho     449: str_Lower(char * __restrict psLine)
                    450: {
                    451:        char *s;
                    452:        register int cx = 0;
                    453: 
                    454:        if (!psLine || !*psLine)
                    455:                return 0;
                    456: 
                    457:        for (s = psLine; *s; s++)
                    458:                if (isupper(*s)) {
                    459:                        *s = tolower(*s);
                    460:                        cx++;
                    461:                }
                    462: 
                    463:        return cx;
                    464: }
1.5       misho     465: 
                    466: /*
1.8       misho     467:  * str_getString() - Get NULL delimited string from data buffer
1.5       misho     468:  *
1.8       misho     469:  * @data = Const data buffer
1.5       misho     470:  * @dlen = Data length
                    471:  * @next = Return next position after string if !=NULL
                    472:  * return: -1 error or size of string 
                    473:  */
                    474: int
                    475: str_getString(const u_char * __restrict data, int dlen, char ** __restrict next)
                    476: {
                    477:        const u_char *pos;
                    478: 
                    479:        if (!data || !dlen)
                    480:                return -1;
                    481: 
                    482:        for (pos = data; pos < data + dlen; pos++)
                    483:                if (!*pos)
                    484:                        break;
                    485:        if (*pos) {
                    486:                elwix_SetErr(ENOEXEC, "Not found null-terminated string");
                    487:                return -1;
                    488:        }
                    489: 
                    490:        if (next)
                    491:                *next = (char*) pos + 1;
                    492:        return pos - data + 1;
1.8       misho     493: }
                    494: 
                    495: /*
                    496:  * str_getString2() - Get string from data buffer with delimiter
                    497:  *
                    498:  * @data = Data buffer
                    499:  * @dlen = Data length
                    500:  * @delim = Data delimiter
                    501:  * @next = Return next position after delimited string if !=NULL
                    502:  * return: -1 error or size of string 
                    503:  */
                    504: int
                    505: str_getString2(char * __restrict data, int dlen, char delim, char ** __restrict next)
                    506: {
                    507:        char *pos;
                    508: 
                    509:        if (!data || !dlen)
                    510:                return -1;
                    511: 
                    512:        for (pos = data; pos < data + dlen; pos++)
                    513:                if (!*pos || *pos == (u_char) delim) {
                    514:                        *pos = 0;
                    515:                        break;
                    516:                }
                    517:        if (*pos) {
                    518:                elwix_SetErr(ENOEXEC, "Not found null-terminated string");
                    519:                return -1;
                    520:        }
                    521: 
                    522:        if (next)
                    523:                *next = (char*) pos + 1;
                    524:        return pos - data;
1.5       misho     525: }
1.11    ! misho     526: 
        !           527: /*
        !           528:  * str_find2replace() - Search find string into data and replace
        !           529:  *
        !           530:  * @data = input string
        !           531:  * @find = search for string
        !           532:  * @replace = replace to string. If it is NULL then deletes found search string
        !           533:  * @str = new produced allocate string. If it is NULL then just return found occurances of find
        !           534:  * @mlen = allocated memory size for new string
        !           535:  * return: -1 error, 0 not found or >0 how many occurances we have for find string
        !           536:  */
        !           537: int
        !           538: str_find2replace(const char *data, const char *find, const char *replace, char **str, int *mlen)
        !           539: {
        !           540:        int cnt = 0, data_len, slen, find_len, replace_len = 0;
        !           541:        const char *pos, *s;
        !           542: 
        !           543:        if (!data || !find)
        !           544:                return -1;
        !           545: 
        !           546:        find_len = strlen(find);
        !           547:        data_len = strlen(data);
        !           548:        for (pos = data; (pos = strstr(pos, find)); pos += find_len, cnt++);
        !           549: 
        !           550:        /* just count occurances */
        !           551:        if (!str || !mlen)
        !           552:                return cnt;
        !           553: 
        !           554:        slen = data_len - find_len * cnt;
        !           555:        if (replace) {
        !           556:                replace_len = strlen(replace);
        !           557:                slen += replace_len * cnt;
        !           558:        }
        !           559:        /* alloc exports string */
        !           560:        *mlen = slen + 1;
        !           561:        *str = e_malloc(*mlen);
        !           562:        if (!*str)
        !           563:                return -1;
        !           564:        else
        !           565:                memset(*str, 0, *mlen);
        !           566: 
        !           567:        /* search & replace */
        !           568:        for (pos = data; *pos;) {
        !           569:                s = strstr(pos, find);
        !           570:                if (s)
        !           571:                        slen = s - pos;
        !           572:                else
        !           573:                        slen = strlen(pos);
        !           574:                /* 
        !           575:                 * This hack was made due to behaviour of compiler against strncat(*str, pos, slen)!
        !           576:                 */
        !           577:                memcpy(*str + strlen(*str), pos, slen);
        !           578:                if (s) {
        !           579:                        if (replace)
        !           580:                                strncat(*str, replace, replace_len);
        !           581:                        pos += find_len;
        !           582:                }
        !           583:                pos += slen;
        !           584:        }
        !           585: 
        !           586:        return cnt;
        !           587: }

FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>