blob: 11059d1a5204ffa87f02dc89a16aa2e46c9333c2 [file] [log] [blame]
Eric Andersenaad1a882001-03-16 22:47:14 +00001/* vi: set sw=4 ts=4: */
2/*
3 * Utility routines.
4 *
Manuel Novoa III cad53642003-03-19 09:13:01 +00005 * Copyright (C) Manuel Novoa III <mjn3@codepoet.org>
Glenn L McGrathb4a1baa2003-01-13 22:09:50 +00006 * and Vladimir Oleynik <dzo@simtreas.ru>
Eric Andersenaad1a882001-03-16 22:47:14 +00007 *
Bernhard Reutner-Fischerb1629b12006-05-19 19:29:19 +00008 * Licensed under GPLv2 or later, see file LICENSE in this tarball for details.
Eric Andersenaad1a882001-03-16 22:47:14 +00009 */
10
Eric Andersenaad1a882001-03-16 22:47:14 +000011#include "libbb.h"
12
Manuel Novoa III 413db4d2004-07-29 23:15:16 +000013#define WANT_HEX_ESCAPES 1
14
15/* Usual "this only works for ascii compatible encodings" disclaimer. */
16#undef _tolower
17#define _tolower(X) ((X)|((char) 0x20))
18
Denis Vlasenkodefc1ea2008-06-27 02:52:20 +000019char FAST_FUNC bb_process_escape_sequence(const char **ptr)
Eric Andersenaad1a882001-03-16 22:47:14 +000020{
Denis Vlasenko248b4a72008-09-19 23:43:59 +000021 /* bash builtin "echo -e '\ec'" interprets \e as ESC,
22 * but coreutils "/bin/echo -e '\ec'" does not.
23 * manpages tend to support coreutils way. */
Denis Vlasenko6ca409e2007-08-12 20:58:27 +000024 static const char charmap[] ALIGN1 = {
Denis Vlasenko248b4a72008-09-19 23:43:59 +000025 'a', 'b', /*'e',*/ 'f', 'n', 'r', 't', 'v', '\\', 0,
26 '\a', '\b', /*27,*/ '\f', '\n', '\r', '\t', '\v', '\\', '\\' };
Eric Andersenaad1a882001-03-16 22:47:14 +000027
Manuel Novoa III 413db4d2004-07-29 23:15:16 +000028 const char *p;
29 const char *q;
Denis Vlasenko248b4a72008-09-19 23:43:59 +000030 unsigned num_digits;
31 unsigned r;
32 unsigned n;
33 unsigned d;
34 unsigned base;
Manuel Novoa III 413db4d2004-07-29 23:15:16 +000035
36 num_digits = n = 0;
37 base = 8;
Eric Andersene5dfced2001-04-09 22:48:12 +000038 q = *ptr;
Eric Andersenaad1a882001-03-16 22:47:14 +000039
Manuel Novoa III 413db4d2004-07-29 23:15:16 +000040#ifdef WANT_HEX_ESCAPES
Eric Andersenb2a30052004-07-26 12:11:32 +000041 if (*q == 'x') {
Eric Andersenb2a30052004-07-26 12:11:32 +000042 ++q;
Manuel Novoa III 413db4d2004-07-29 23:15:16 +000043 base = 16;
44 ++num_digits;
Eric Andersenb2a30052004-07-26 12:11:32 +000045 }
Manuel Novoa III 413db4d2004-07-29 23:15:16 +000046#endif
Eric Andersenb2a30052004-07-26 12:11:32 +000047
Denys Vlasenkoecc2a2e2009-08-29 22:53:41 +020048 /* bash requires leading 0 in octal escapes:
49 * \02 works, \2 does not (prints \ and 2).
50 * We treat \2 as a valid octal escape sequence. */
Manuel Novoa III cad53642003-03-19 09:13:01 +000051 do {
Denis Vlasenko7e754f12007-04-09 13:04:50 +000052 d = (unsigned char)(*q) - '0';
Manuel Novoa III 413db4d2004-07-29 23:15:16 +000053#ifdef WANT_HEX_ESCAPES
54 if (d >= 10) {
Denis Vlasenko7e754f12007-04-09 13:04:50 +000055 d = (unsigned char)(_tolower(*q)) - 'a' + 10;
Manuel Novoa III 413db4d2004-07-29 23:15:16 +000056 }
57#endif
58
59 if (d >= base) {
60#ifdef WANT_HEX_ESCAPES
61 if ((base == 16) && (!--num_digits)) {
Tim Rikerc1ef7bd2006-01-25 00:08:53 +000062/* return '\\'; */
Manuel Novoa III 413db4d2004-07-29 23:15:16 +000063 --q;
64 }
65#endif
Eric Andersenccfc4482004-07-27 16:45:46 +000066 break;
Eric Andersenb2a30052004-07-26 12:11:32 +000067 }
Manuel Novoa III 413db4d2004-07-29 23:15:16 +000068
69 r = n * base + d;
70 if (r > UCHAR_MAX) {
71 break;
Eric Andersene5dfced2001-04-09 22:48:12 +000072 }
Manuel Novoa III 413db4d2004-07-29 23:15:16 +000073
74 n = r;
75 ++q;
76 } while (++num_digits < 3);
Eric Andersenaad1a882001-03-16 22:47:14 +000077
Eric Andersene5dfced2001-04-09 22:48:12 +000078 if (num_digits == 0) { /* mnemonic escape sequence? */
Manuel Novoa III cad53642003-03-19 09:13:01 +000079 p = charmap;
80 do {
Eric Andersene5dfced2001-04-09 22:48:12 +000081 if (*p == *q) {
82 q++;
83 break;
84 }
Manuel Novoa III cad53642003-03-19 09:13:01 +000085 } while (*++p);
Denys Vlasenkoecc2a2e2009-08-29 22:53:41 +020086 /* p points to found escape char or NUL,
87 * advance it and find what it translates to */
88 p += sizeof(charmap) / 2;
89 n = *p;
Eric Andersene5dfced2001-04-09 22:48:12 +000090 }
Eric Andersenaad1a882001-03-16 22:47:14 +000091
Eric Andersene5dfced2001-04-09 22:48:12 +000092 *ptr = q;
Manuel Novoa III 413db4d2004-07-29 23:15:16 +000093
Eric Andersene5dfced2001-04-09 22:48:12 +000094 return (char) n;
Eric Andersenaad1a882001-03-16 22:47:14 +000095}