| /* base64.c - Encode and decode base64 |
| * |
| * Copyright 2014 Rob Landley <rob@landley.net> |
| * |
| * No standard |
| |
| USE_BASE64(NEWTOY(base64, "diw#<0=76[!dw]", TOYFLAG_USR|TOYFLAG_BIN)) |
| |
| config BASE64 |
| bool "base64" |
| default y |
| help |
| usage: base64 [-di] [-w COLUMNS] [FILE...] |
| |
| Encode or decode in base64. |
| |
| -d decode |
| -i ignore non-alphabetic characters |
| -w wrap output at COLUMNS (default 76 or 0 for no wrap) |
| */ |
| |
| #define FOR_base64 |
| #include "toys.h" |
| |
| GLOBALS( |
| long columns; |
| |
| unsigned total; |
| ) |
| |
| static void wraputchar(int c, int *x) |
| { |
| putchar(c); |
| TT.total++; |
| if (TT.columns && ++*x == TT.columns) { |
| *x = 0; |
| xputc('\n'); |
| }; |
| } |
| |
| static void do_base64(int fd, char *name) |
| { |
| int out = 0, bits = 0, x = 0, i, len; |
| char *buf = toybuf+128; |
| |
| TT.total = 0; |
| |
| for (;;) { |
| // If no more data, flush buffer |
| if (!(len = xread(fd, buf, sizeof(toybuf)-128))) { |
| if (!(toys.optflags & FLAG_d)) { |
| if (bits) wraputchar(toybuf[out<<(6-bits)], &x); |
| while (TT.total&3) wraputchar('=', &x); |
| if (x) xputc('\n'); |
| } |
| |
| return; |
| } |
| |
| for (i=0; i<len; i++) { |
| if (toys.optflags & FLAG_d) { |
| if (buf[i] == '=') return; |
| |
| if ((x = stridx(toybuf, buf[i])) != -1) { |
| out = (out<<6) + x; |
| bits += 6; |
| if (bits >= 8) { |
| putchar(out >> (bits -= 8)); |
| out &= (1<<bits)-1; |
| if (ferror(stdout)) perror_exit(0); |
| } |
| |
| continue; |
| } |
| if (buf[i] == '\n' || (toys.optflags & FLAG_i)) continue; |
| |
| break; |
| } else { |
| out = (out<<8) + buf[i]; |
| bits += 8; |
| while (bits >= 6) { |
| wraputchar(toybuf[out >> (bits -= 6)], &x); |
| out &= (1<<bits)-1; |
| } |
| } |
| } |
| } |
| } |
| |
| void base64_main(void) |
| { |
| base64_init(toybuf); |
| loopfiles(toys.optargs, do_base64); |
| } |