base64.c (11537B)
- // utils-std: Collection of commonly available Unix tools
- // SPDX-FileCopyrightText: 2017 Haelwenn (lanodan) Monnier <contact+utils@hacktivis.me>
- // SPDX-FileCopyrightText: 2018 The NetBSD Foundation, Inc.
- // SPDX-License-Identifier: MPL-2.0 AND BSD-2-Clause
- #define _POSIX_C_SOURCE 200809L
- #include "../lib/getopt_nolong.h"
- #include <assert.h> /* assert */
- #include <ctype.h> /* isspace */
- #include <errno.h> /* errno */
- #include <stdint.h> /* uint8_t */
- #include <stdio.h> /* fopen(), fprintf() */
- #include <stdlib.h> /* abort */
- #include <string.h> /* strerror(), strncmp() */
- #include <sys/stat.h> /* fstat */
- #include <unistd.h> /* read(), write(), close(), getopt() */
- // 64(26+26+10+2) + NULL
- static const char *b64_encmap = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
- static size_t c_out = 0;
- // 76 is lowest of all base64 related RFCs
- static long wrap_nl = 76;
- const char *argv0 = "base64";
- static int
- xputc(int c, FILE *stream)
- {
- if(fputc(c, stream) == EOF)
- {
- fprintf(stderr, "%s: error: Failed writing: %s\n", argv0, strerror(errno));
- errno = 0;
- return 1;
- }
- int err = ferror(stream);
- if(err != 0)
- {
- fprintf(stderr, "%s: error: Failed writing: %s\n", argv0, strerror(err));
- errno = 0;
- return 1;
- }
- return 0;
- }
- static inline uint8_t
- b64_get(uint8_t pos)
- {
- assert(pos <= 64);
- return b64_encmap[pos];
- }
- static void
- b64encode(uint8_t out[4], uint8_t in[3])
- {
- out[0] = b64_get(in[0] >> 2);
- out[1] = b64_get((uint8_t)(((in[0] & 0x03) << 4) | (in[1] >> 4)));
- out[2] = b64_get((uint8_t)(((in[1] & 0x0f) << 2) | (in[2] >> 6)));
- out[3] = b64_get(in[2] & 0x3f);
- }
- static int
- base64_encode(FILE *fin, const char *iname)
- {
- while(1)
- {
- uint8_t obuf[4] = "----";
- uint8_t ibuf[3] = {0, 0, 0};
- uint8_t pad = 0;
- size_t c = 0;
- for(; c < 3; c++)
- {
- int buf = getc(fin);
- if(buf == EOF)
- {
- break;
- }
- ibuf[c] = buf;
- }
- if(c == 0)
- {
- return 0;
- };
- if(ferror(fin))
- {
- fprintf(
- stderr, "%s: error: Failed reading from file '%s': %s\n", argv0, iname, strerror(errno));
- errno = 0;
- return 1;
- }
- for(; c < 3; pad++, c++)
- {
- ibuf[c] = 0;
- }
- assert(c == 3);
- assert(pad <= 3);
- b64encode(obuf, (uint8_t *)ibuf);
- for(; pad > 0; pad--)
- {
- obuf[4 - pad] = '=';
- }
- if(fwrite((char *)obuf, 4, 1, stdout) <= 0)
- {
- fprintf(stderr, "%s: error: Failed writing: %s\n", argv0, strerror(errno));
- errno = 0;
- return 1;
- }
- c_out += 4;
- if(wrap_nl != 0 && (c_out + 4) > wrap_nl)
- {
- c_out = 0;
- if(xputc('\n', stdout) != 0) return 1;
- }
- if(feof(fin))
- {
- if(fflush(stdout))
- {
- fprintf(stderr, "%s: error: Failed writing: %s\n", argv0, strerror(errno));
- errno = 0;
- return 1;
- }
- int err = ferror(stdout);
- if(err != 0)
- {
- fprintf(stderr, "%s: error: Failed writing: %s\n", argv0, strerror(errno));
- return 1;
- }
- return 0;
- }
- }
- abort(); // unreachable
- }
- // This function is based on NetBSD's code, which contains the following notices:
- // Copyright (c) 2018 The NetBSD Foundation, Inc.
- //
- // This code is derived from software contributed to The NetBSD Foundation
- // by Christos Zoulas.
- static int
- base64_decode(FILE *fin, const char *iname)
- {
- int c = 0;
- uint8_t out = 0;
- int state = 0;
- while((c = getc(fin)) != EOF)
- {
- if(isspace(c)) continue;
- if(c == '=') break;
- uint8_t b = 65;
- for(uint8_t i = 0; i < 64; i++)
- {
- if(b64_encmap[i] == c)
- {
- b = i;
- break;
- }
- }
- if(b > 64)
- {
- fprintf(stderr, "%s: error: Invalid character '%c'\n", argv0, c);
- return 1;
- }
- //fprintf(stderr, "state: %d | c: %c (%d) | b: %d\n", state, c, c, b);
- switch(state)
- {
- case 0:
- out = (uint8_t)(b << 2);
- break;
- case 1:
- out |= b >> 4;
- if(xputc(out, stdout) != 0) return 1;
- out = (uint8_t)((b & 0xf) << 4);
- break;
- case 2:
- out |= b >> 2;
- if(xputc(out, stdout) != 0) return 1;
- out = (uint8_t)((b & 0x3) << 6);
- break;
- case 3:
- out |= b;
- if(xputc(out, stdout) != 0) return 1;
- out = 0;
- break;
- default:
- abort();
- }
- state = (state + 1) & 3;
- }
- //fprintf(stderr, "[outside] state: %d | c: %c (%d) | out: %d\n", state, c, c, out);
- if(c == '=')
- {
- switch(state)
- {
- case 0:
- case 1:
- fprintf(stderr, "%s: error: Invalid character '%c' (early '=')\n", argv0, c);
- return 1;
- case 2:
- while(isspace(c = getc(fin)))
- ;
- if(c != '=')
- {
- fprintf(stderr, "%s: error: Invalid character '%c' (not '=')\n", argv0, c);
- return 1;
- }
- /* fallthrough */
- case 3:
- while(isspace(c = getc(fin)))
- ;
- if(c != EOF)
- {
- fprintf(stderr, "%s: error: Invalid character '%c' (not EOF)\n", argv0, c);
- return 1;
- }
- return 0;
- default:
- abort();
- }
- }
- assert(c == EOF || state == 0);
- if(fflush(stdout))
- {
- fprintf(stderr, "%s: error: Failed writing: %s\n", argv0, strerror(errno));
- errno = 0;
- return 1;
- }
- int err = ferror(stdout);
- if(err != 0)
- {
- fprintf(stderr, "%s: error: Failed writing: %s\n", argv0, strerror(err));
- errno = 0;
- return 1;
- }
- return 0;
- }
- static int
- base64_main(int argc, char *argv[])
- {
- argv0 = "base64";
- int (*process)(FILE *, const char *) = &base64_encode;
- int ret = 0;
- for(int c = -1; (c = getopt_nolong(argc, argv, ":dw:")) != -1;)
- {
- switch(c)
- {
- case 'd':
- process = &base64_decode;
- break;
- case 'w':
- errno = 0;
- char *e = NULL;
- wrap_nl = strtol(optarg, &e, 10);
- // extraneous characters is invalid
- if(e && *e != 0) errno = EINVAL;
- if(errno != 0)
- {
- fprintf(stderr, "%s: error: Option '-w %s': %s\n", argv0, optarg, strerror(errno));
- return 1;
- }
- break;
- case ':':
- fprintf(stderr, "%s: error: Missing operand for option '-%c'\n", argv0, optopt);
- return 1;
- case '?':
- if(!got_long_opt) fprintf(stderr, "%s: error: Unrecognised option '-%c'\n", argv0, optopt);
- return 1;
- }
- }
- argc -= optind;
- argv += optind;
- if(argc <= 0)
- {
- ret = process(stdin, "<stdin>");
- goto end;
- }
- for(int argi = 0; argi < argc; argi++)
- {
- if(strncmp(argv[argi], "-", 2) == 0)
- {
- if(process(stdin, "<stdin>") != 0)
- {
- ret = 1;
- goto end;
- }
- }
- else if(strncmp(argv[argi], "--", 3) == 0)
- {
- continue;
- }
- else
- {
- FILE *fin = fopen(argv[argi], "r");
- if(fin == NULL || ferror(fin) != 0)
- {
- fprintf(stderr,
- "%s: error: Failed opening file '%s': %s\n",
- argv0,
- argv[argi],
- strerror(errno));
- ret = 1;
- goto end;
- }
- if(process(fin, argv[argi]) != 0) ret = 1;
- if(fclose(fin) < 0)
- {
- fprintf(stderr,
- "%s: error: Failed closing file '%s': %s\n",
- argv0,
- argv[argi],
- strerror(errno));
- ret = 1;
- goto end;
- }
- }
- }
- end:
- if(wrap_nl != 0 && c_out > 0)
- {
- printf("\n");
- }
- if(fclose(stdin) != 0)
- {
- fprintf(stderr, "%s: error: Failed closing file <stdin>: %s\n", argv0, strerror(errno));
- ret = 1;
- }
- if(fclose(stdout) != 0)
- {
- fprintf(stderr, "%s: error: Failed closing file <stdout>: %s\n", argv0, strerror(errno));
- ret = 1;
- }
- return ret;
- }
- static int
- uuencode(FILE *fin, const char *iname)
- {
- while(1)
- {
- char output[60] = "";
- size_t pos = 0;
- int len = 0;
- for(; len < 45; pos += 4, len += 3)
- {
- uint8_t ibuf[3] = {0, 0, 0};
- size_t c = 0;
- for(; c < 3; c++)
- {
- int buf = getc(fin);
- if(buf == EOF)
- {
- break;
- }
- ibuf[c] = buf;
- }
- if(c == 0)
- {
- break;
- }
- if(ferror(fin))
- {
- fprintf(stderr,
- "%s: error: Failed reading from file '%s': %s\n",
- argv0,
- iname,
- strerror(errno));
- errno = 0;
- return 1;
- }
- assert((3 + pos) < 60);
- /* conversion math taken from POSIX.1-2024 specification */
- /* clang-format off */
- output[0+pos] = 0x20 + (( ibuf[0] >> 2 ) & 0x3F);
- output[1+pos] = 0x20 + (((ibuf[0] << 4) | ((ibuf[1] >> 4) & 0x0F)) & 0x3F);
- output[2+pos] = 0x20 + (((ibuf[1] << 2) | ((ibuf[2] >> 6) & 0x03)) & 0x3F);
- output[3+pos] = 0x20 + (( ibuf[2] ) & 0x3F);
- /* clang-format on */
- for(int i = 0; i < 4; i++)
- if(output[i + pos] == 0x20) output[i + pos] = 0x60;
- }
- if(printf("%c", len == 0 ? 0x60 : 0x20 + len) < 0)
- {
- fprintf(stderr, "%s: error: Failed writing length: %s\n", argv0, strerror(errno));
- errno = 0;
- return 1;
- }
- if(pos > 0)
- if(fwrite(output, pos, 1, stdout) <= 0)
- {
- fprintf(stderr, "%s: error: Failed writing data: %s\n", argv0, strerror(errno));
- errno = 0;
- return 1;
- }
- if(printf("\n") < 0)
- {
- fprintf(stderr, "%s: error: Failed writing newline: %s\n", argv0, strerror(errno));
- errno = 0;
- return 1;
- }
- if(feof(fin)) break;
- }
- if(fflush(stdout) != 0)
- {
- fprintf(stderr, "%s: error: Failed writing (flush): %s\n", argv0, strerror(errno));
- errno = 0;
- return 1;
- }
- int err = ferror(stdout);
- if(err != 0)
- {
- fprintf(stderr, "%s: error: Failed writing (ferror): %s\n", argv0, strerror(errno));
- return 1;
- }
- return 0;
- }
- static int
- uuencode_main(int argc, char *argv[])
- {
- argv0 = "uuencode";
- const char *begin_str = "begin";
- const char *end_str = "end\n";
- int (*process)(FILE *, const char *) = &uuencode;
- // get old, then reset to old. Yay to POSIX nonsense
- mode_t mask = umask(0);
- umask(mask);
- // negate to get a normal bitmask
- mask ^= 0777;
- wrap_nl = 45;
- int ret = 0;
- for(int c = -1; (c = getopt_nolong(argc, argv, ":m")) != -1;)
- {
- switch(c)
- {
- case 'm':
- process = &base64_encode;
- begin_str = "begin-base64";
- end_str = "====\n";
- wrap_nl = 76;
- break;
- case ':':
- fprintf(stderr, "%s: error: Missing operand for option '-%c'\n", argv0, optopt);
- return 1;
- case '?':
- if(!got_long_opt) fprintf(stderr, "%s: error: Unrecognised option '-%c'\n", argv0, optopt);
- return 1;
- }
- }
- argc -= optind;
- argv += optind;
- if(argc > 2)
- {
- fprintf(stderr, "%s: error: Expected 2 arguments or less, got %d\n", argv0, argc);
- return 1;
- }
- FILE *fin = stdin;
- const char *decode_path = "-";
- const char *iname = "<stdin>";
- mode_t mode = 0666 & mask;
- if(argc >= 1) decode_path = argv[argc - 1];
- if(argc == 2)
- {
- iname = argv[0];
- fin = fopen(iname, "rb");
- if(!fin)
- {
- fprintf(
- stderr, "%s: error: Failed opening input file '%s': %s\n", argv0, iname, strerror(errno));
- return 1;
- }
- struct stat fin_stat;
- if(fstat(fileno(fin), &fin_stat) != 0)
- {
- fprintf(stderr,
- "%s: warning: Failed getting status for file '%s': %s\n",
- argv0,
- iname,
- strerror(errno));
- ret = 1;
- }
- mode = fin_stat.st_mode & 0777;
- }
- printf("%s %03o %s\n", begin_str, mode, decode_path);
- if(process(fin, iname) != 0)
- {
- ret = 1;
- goto end;
- }
- end:
- if(ret == 0)
- {
- if(wrap_nl != 0 && c_out > 0) printf("\n");
- printf("%s", end_str);
- }
- if(fclose(fin) != 0)
- {
- fprintf(stderr, "%s: error: Failed closing file \"%s\": %s\n", argv0, iname, strerror(errno));
- ret = 1;
- }
- if(fclose(stdout) != 0)
- {
- fprintf(stderr, "%s: error: Failed closing file <stdout>: %s\n", argv0, strerror(errno));
- ret = 1;
- }
- return ret;
- }
- int
- main(int argc, char *argv[])
- {
- const char *arg0 = argv[0];
- const char *s = strrchr(arg0, '/');
- if(s) arg0 = s + 1;
- if(strcmp(arg0, "base64") == 0) return base64_main(argc, argv);
- if(strcmp(arg0, "uuencode") == 0) return uuencode_main(argc, argv);
- fprintf(stderr, "%s: error: Unknown utility '%s' expected 'base64' or 'uuencode'\n", arg0, arg0);
- return -1;
- }