logo

utils-std

Collection of commonly available Unix tools git clone https://anongit.hacktivis.me/git/utils-std.git/
commit: 80fc05ca12019b1145e407b921cd03cf6162c471
parent b1ce552adedbaa4b74341a1635fe12111aabbee9
Author: Haelwenn (lanodan) Monnier <contact@hacktivis.me>
Date:   Wed, 26 Mar 2025 14:35:22 +0100

cmd/echo: support -e and -E options for toggling escape codes support

Diffstat:

Mcmd/echo.139++++++++++++++++++++++++++++++++++-----
Mcmd/echo.c116+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++----
Mtest-cmd/echo.sh41+++++++++++++++++++++++++++++++++++++++--
3 files changed, 184 insertions(+), 12 deletions(-)

diff --git a/cmd/echo.1 b/cmd/echo.1 @@ -9,7 +9,7 @@ .Nd write arguments to standard output .Sh SYNOPSIS .Nm -.Op Fl n +.Op Fl Een .Op Ar string... .Sh DESCRIPTION .Nm @@ -28,16 +28,45 @@ operand, which generally terminates option processing, is treated as part of .Bl -tag -width Ds .It Fl n Do not print the trailing newline character. +.It Fl E +Toggle off the support for escape codes. (default) +.It Fl e +Toggle on support for the following escape codes: +.Bl -tag -compact -width _a +.It \ea +Write an <alert>. +.It \eb +Write a <backspace>. +.It \ec +Clear the rest of the output, including the trailing newline. +.It \ef +Write a <form-feed>. +.It \en +Write a <newline>. +.It \er +Write a <carriage-return>. +.It \et +Write a <tab>. +.It \ev +Write a <vertical-tab>. +.It \e\e +Write a <backslash> character. +.It \e0 Ns Ar num +Write an octet corresponding to the zero, one, two, or three digits octal number +.Ar num . +For example +.Ql \e0 +writes the NULL byte, and +.Ql \e01 +writes a octet of value 1. +.El .El .Sh EXIT STATUS .Ex -std .Sh SEE ALSO .Xr printf 1 .Sh STANDARDS -Not XSI-compliant as -.Fl n -is parsed as an option and backslash operators aren't supported. -Should be compliant with the rest of the +Should be compliant with the IEEE Std 1003.1-2024 (“POSIX.1”) specification. .Sh AUTHORS diff --git a/cmd/echo.c b/cmd/echo.c @@ -5,6 +5,7 @@ #define _POSIX_C_SOURCE 200809L #include <stdbool.h> #include <stdio.h> // perror +#include <stdlib.h> // malloc #include <string.h> // strlen #include <unistd.h> // write @@ -12,14 +13,37 @@ int main(int argc, char *argv[]) { size_t arg_len = 0; - bool opt_n = false; + bool opt_n = false, opt_e = false; argc--; argv++; - if(argc > 0 && strncmp(*argv, "-n", 3) == 0) + for(; argc > 0;) { - opt_n = true; + if(argv[0][0] != '-') break; + + /* '--' needs to be passed as-is so no argc--,argv++ */ + if(argv[0][1] == '-') break; + + for(int i = 1; argv[0][i] != '\0'; i++) + { + switch(argv[0][i]) + { + case 'n': + opt_n = true; + break; + case 'e': + opt_e = true; + break; + case 'E': + opt_e = false; + break; + default: + fprintf(stderr, "echo: warning: unknown option '-%c'\n", argv[0][i]); + break; + } + } + argc--; argv++; } @@ -48,12 +72,94 @@ main(int argc, char *argv[]) if(opt_n) arg_len--; // no newline - ssize_t nwrite = write(1, *argv, arg_len); - if(nwrite < (ssize_t)arg_len) + char *d = *argv; + size_t d_len = arg_len; + + char *newd = NULL; + if(opt_e) + { + newd = malloc(arg_len); + if(!newd) + { + fprintf(stderr, "echo: error: Failed to allocate for a copy of the strings\n"); + return 1; + } + + int di = 0; + + for(size_t argi = 0; argi < arg_len; argi++) + { + if(d[argi] != '\\') + { + newd[di++] = d[argi]; + continue; + } + + switch(d[++argi]) + { + case 'a': + newd[di++] = '\a'; + break; + case 'b': + newd[di++] = '\b'; + break; + case 'c': + newd[di++] = '\0'; + goto p_break; + case 'f': + newd[di++] = '\f'; + break; + case 'n': + newd[di++] = '\n'; + break; + case 'r': + newd[di++] = '\r'; + break; + case 't': + newd[di++] = '\t'; + break; + case 'v': + newd[di++] = '\v'; + break; + case '\\': + newd[di++] = '\\'; + break; + case '0': /* \0 \0n \0nn \0nnn */ + int nl = 1; // skip leading 0 + int num = 0; + for(; nl < 4; nl++) + { + if(d[argi + nl] >= '0' && d[argi + nl] <= '7') + { + num = (num * 8) + (d[argi + nl] - '0'); + } + else + break; + } + newd[di++] = num; + argi += (nl - 1); + break; + default: + newd[di++] = d[argi]; + break; + } + } + + newd[di] = '\0'; + p_break: + d = newd; + d_len = di; + } + + ssize_t nwrite = write(1, d, d_len); + if(nwrite < (ssize_t)d_len) { perror("echo: error: Failed writing"); + if(opt_e) free(newd); return 1; } + if(opt_e) free(newd); + return 0; } diff --git a/test-cmd/echo.sh b/test-cmd/echo.sh @@ -2,8 +2,9 @@ # SPDX-FileCopyrightText: 2017 Haelwenn (lanodan) Monnier <contact+utils@hacktivis.me> # SPDX-License-Identifier: MPL-2.0 -target="$(dirname "$0")/../cmd/echo" -plans=7 +WD="$(dirname "$0")/../" +target="${WD}/cmd/echo" +plans=17 . "$(dirname "$0")/tap.sh" t 'empty' '' ' @@ -17,3 +18,39 @@ t -- '-n' '-n' '' t -- '-n foo' '-n foo' 'foo' t -- '-n foo bar' '-n foo bar' 'foo bar' t -- '-n -- foo' '-n -- foo' '-- foo' + +wrap_od() { + "$target" "$@" | od -Ax -t x1 | tr '[:lower:]' '[:upper:]' +} + +t_args 'e:newline' 'foo + +' -e 'foo\n' + +t_args 'en:newline' 'foo +' -en 'foo\n' + +t_args 'e:simple_esc' "$(printf %b '\a\b\f\n\r\t\v') +" -e '\a\b\f\n\r\t\v' + +t_args 'e:clear' 'foo' -e 'foo\cbar' 'baz' + +t_cmd 'e:0' '000000 66 6F 6F 00 62 61 72 00 +000008 +' wrap_od -en 'foo\0bar\0' + +# od -An -t o1 test-cmd/inputs/all_bytes | sed 's; ;\\0;g' | tr -d '\n' +all_bytes_octal='\0000\0001\0002\0003\0004\0005\0006\0007\0010\0011\0012\0013\0014\0015\0016\0017\0020\0021\0022\0023\0024\0025\0026\0027\0030\0031\0032\0033\0034\0035\0036\0037\0040\0041\0042\0043\0044\0045\0046\0047\0050\0051\0052\0053\0054\0055\0056\0057\0060\0061\0062\0063\0064\0065\0066\0067\0070\0071\0072\0073\0074\0075\0076\0077\0100\0101\0102\0103\0104\0105\0106\0107\0110\0111\0112\0113\0114\0115\0116\0117\0120\0121\0122\0123\0124\0125\0126\0127\0130\0131\0132\0133\0134\0135\0136\0137\0140\0141\0142\0143\0144\0145\0146\0147\0150\0151\0152\0153\0154\0155\0156\0157\0160\0161\0162\0163\0164\0165\0166\0167\0170\0171\0172\0173\0174\0175\0176\0177\0200\0201\0202\0203\0204\0205\0206\0207\0210\0211\0212\0213\0214\0215\0216\0217\0220\0221\0222\0223\0224\0225\0226\0227\0230\0231\0232\0233\0234\0235\0236\0237\0240\0241\0242\0243\0244\0245\0246\0247\0250\0251\0252\0253\0254\0255\0256\0257\0260\0261\0262\0263\0264\0265\0266\0267\0270\0271\0272\0273\0274\0275\0276\0277\0300\0301\0302\0303\0304\0305\0306\0307\0310\0311\0312\0313\0314\0315\0316\0317\0320\0321\0322\0323\0324\0325\0326\0327\0330\0331\0332\0333\0334\0335\0336\0337\0340\0341\0342\0343\0344\0345\0346\0347\0350\0351\0352\0353\0354\0355\0356\0357\0360\0361\0362\0363\0364\0365\0366\0367\0370\0371\0372\0373\0374\0375\0376\0377' +t_file 'en:all_bytes' "$WD"/test-cmd/inputs/all_bytes -en "$all_bytes_octal" + +t_args 'E:newline' 'foo\n +' -E 'foo\n' + +t_args 'E:simple_esc' '\a\b\f\n\r\t\v +' -E '\a\b\f\n\r\t\v' + +t_args 'E:clear' 'foo\cbar baz +' -E 'foo\cbar' 'baz' + +t_args 'E:0' 'foo\0bar\0 +' -E 'foo\0bar\0'