aboutsummaryrefslogtreecommitdiff
path: root/toys/echo.c
blob: f7c423cf01b06adf758abda8b119d30283ec4fc6 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
/* vi: set sw=4 ts=4:
 *
 * echo.c - echo supporting -n and -e.
 *
 * Copyright 2007 Rob Landley <rob@landley.net>
 *
 * See http://www.opengroup.org/onlinepubs/009695399/utilities/echo.html

USE_ECHO(NEWTOY(echo, "^?en", TOYFLAG_BIN))

config ECHO
	bool "echo"
	default y
	help
	  usage: echo [-ne] [args...]

	  Write each argument to stdout, with one space between each, followed
	  by a newline.

	  -n    No trailing newline.
	  -e    Process the following escape sequences:
	   \\     backslash
	   \0NNN octal values (1 to 3 digits)
	   \a    alert (beep/flash)
	   \b    backspace
	   \c    stop output here (avoids trailing newline)
	   \f    form feed
	   \n    newline
	   \r    carriage return
	   \t    horizontal tab
	   \v    vertical tab
	   \xHH  hexadecimal values (1 to 2 digits)
*/

#include "toys.h"

void echo_main(void)
{
	int i = 0;
	char *arg, *from = "\\abfnrtv", *to = "\\\a\b\f\n\r\t\v";

	for (;;) {
		arg = toys.optargs[i];
		if (!arg) break;
		if (i++) xputc(' ');

		// Handle -e

		if (toys.optflags&2) {
			int c, j = 0;
			for (;;) {
				c = arg[j++];
				if (!c) break;
				if (c == '\\') {
					char *found;
					int d = arg[j++];

					// handle \escapes

					if (d) {
						found = strchr(from, d);
						if (found) c = to[found-from];
						else if (d == 'c') goto done;
						else if (d == '0') {
							int n = 0;
							c = 0;
							while (arg[j]>='0' && arg[j]<='7' && n++<3)
								c = (c*8)+arg[j++]-'0';
						}
						else if (d == 'x') {
							int n = 0;
							c = 0;							
							while (n++<2)
								if (arg[j]>='0' && arg[j]<='9')
									c = (c*16)+arg[j++]-'0';
								else if (tolower(arg[j])>='a' && tolower(arg[j])<='f')
									c = (c*16)+tolower(arg[j++])-'a'+10;
								else
									break;							
						}
					}
				}
				xputc(c);
			}
		} else xprintf("%s", arg);
	}

	// Output "\n" if no -n
	if (!(toys.optflags&1)) xputc('\n');
done:
	xflush();
}