/* SPDX-License-Identifier: GPL-3.0-or-later */ /* * Copyright (C) 2018 - David Oberhollenzer * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include #include #include #include "util.h" static int xdigit(int x) { if (isupper(x)) return x - 'A' + 0x0A; if (islower(x)) return x - 'a' + 0x0A; return x - '0'; } int unescape(char *src) { char *dst = src; int c; for (;;) { while (*src != '"' && *src != '\0') *(dst++) = *(src++); if (*src == '\0') break; ++src; while ((c = *(src++)) != '"') { if (c == '\0') return -1; if (c == '\\') { c = *(src++); switch (c) { case 'a': c = '\a'; break; case 'b': c = '\b'; break; case 'f': c = '\f'; break; case 'n': c = '\n'; break; case 't': c = '\t'; break; case '\\': case '"': break; case 'x': c = 0; if (isxdigit(*src)) c = (c<<4) | xdigit(*(src++)); if (isxdigit(*src)) c = (c<<4) | xdigit(*(src++)); break; case '0': c = 0; if (isdigit(*src) && *src < '8') c = (c<<3) | (*(src++) - '0'); if (isdigit(*src) && *src < '8') c = (c<<3) | (*(src++) - '0'); if (isdigit(*src) && *src < '8') c = (c<<3) | (*(src++) - '0'); break; default: return -1; } } *(dst++) = c; } } *(dst++) = '\0'; return 0; } char **split_argv(char *str) { size_t i = 0, cap = 0, new_cap; char **argv = NULL, **new; char *ptr; ptr = str; for (;;) { if (*ptr == ' ') { ++ptr; continue; } if (i == cap) { new_cap = cap ? cap * 2 : 16; new = realloc(argv, sizeof(argv[0]) * new_cap); if (new == NULL) { free(argv); errno = ENOMEM; return NULL; } cap = new_cap; argv = new; } if (*ptr == '\0') { argv[i++] = NULL; break; } argv[i++] = ptr; if (*ptr == '"') { ++ptr; while (*ptr != '\0' && *ptr != '"') { if (ptr[0] == '\\' && ptr[1] != '\0') ++ptr; ++ptr; } if (*ptr == '"') ++ptr; if (*ptr == ' ') { *(ptr++) = '\0'; } else if (*ptr != '\0') { goto fail_str; } if (unescape(argv[i - 1])) goto fail_str; } else { while (*ptr != '\0' && *ptr != ' ') ++ptr; if (*ptr == ' ') *(ptr++) = '\0'; } } return argv; fail_str: free(argv); errno = EINVAL; return NULL; }