2015-01-29 20:52:44 +00:00
|
|
|
/* See LICENSE file for copyright and license details. */
|
2017-02-04 23:44:35 +00:00
|
|
|
#include <ctype.h>
|
2015-01-29 20:52:44 +00:00
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "../util.h"
|
|
|
|
|
2017-02-04 23:44:35 +00:00
|
|
|
#define is_odigit(c) ('0' <= c && c <= '7')
|
|
|
|
|
2015-01-29 20:52:44 +00:00
|
|
|
size_t
|
|
|
|
unescape(char *s)
|
|
|
|
{
|
2017-02-04 23:44:35 +00:00
|
|
|
static const char escapes[256] = {
|
|
|
|
['"'] = '"',
|
|
|
|
['\''] = '\'',
|
|
|
|
['\\'] = '\\',
|
|
|
|
['a'] = '\a',
|
|
|
|
['b'] = '\b',
|
|
|
|
['E'] = 033,
|
|
|
|
['e'] = 033,
|
|
|
|
['f'] = '\f',
|
|
|
|
['n'] = '\n',
|
|
|
|
['r'] = '\r',
|
|
|
|
['t'] = '\t',
|
|
|
|
['v'] = '\v'
|
|
|
|
};
|
|
|
|
size_t m, q;
|
|
|
|
char *r, *w;
|
2015-01-29 20:52:44 +00:00
|
|
|
|
2017-02-04 23:44:35 +00:00
|
|
|
for (r = w = s; *r;) {
|
|
|
|
if (*r != '\\') {
|
|
|
|
*w++ = *r++;
|
2015-01-29 20:52:44 +00:00
|
|
|
continue;
|
2017-02-04 23:44:35 +00:00
|
|
|
}
|
|
|
|
r++;
|
|
|
|
if (!*r) {
|
2017-02-04 21:15:13 +00:00
|
|
|
eprintf("null escape sequence\n");
|
2017-02-04 23:44:35 +00:00
|
|
|
} else if (escapes[(unsigned char)*r]) {
|
|
|
|
*w++ = escapes[(unsigned char)*r++];
|
|
|
|
} else if (is_odigit(*r)) {
|
|
|
|
for (q = 0, m = 4; m && is_odigit(*r); m--, r++)
|
|
|
|
q = q * 8 + (*r - '0');
|
|
|
|
*w++ = MIN(q, 255);
|
|
|
|
} else if (*r == 'x' && isxdigit(r[1])) {
|
|
|
|
r++;
|
|
|
|
for (q = 0, m = 2; m && isxdigit(*r); m--, r++)
|
|
|
|
if (isdigit(*r))
|
|
|
|
q = q * 16 + (*r - '0');
|
|
|
|
else
|
|
|
|
q = q * 16 + (tolower(*r) - 'a' + 10);
|
|
|
|
*w++ = q;
|
|
|
|
} else {
|
|
|
|
eprintf("invalid escape sequence '\\%c'\n", *r);
|
2015-01-29 20:52:44 +00:00
|
|
|
}
|
|
|
|
}
|
2017-03-08 07:10:48 +00:00
|
|
|
*w = '\0';
|
2015-01-29 20:52:44 +00:00
|
|
|
|
2017-02-04 23:44:35 +00:00
|
|
|
return w - s;
|
2015-01-29 20:52:44 +00:00
|
|
|
}
|