diff options
author | Mattias Andrée <m@maandree.se> | 2025-03-02 18:38:02 +0100 |
---|---|---|
committer | Mattias Andrée <m@maandree.se> | 2025-03-02 18:38:02 +0100 |
commit | b60d205c290e843e05009cc709f2d3d1c1cd4aea (patch) | |
tree | 580efa2c4965d6bdd83f450d341136f7458a4db8 /lss16toppm.c | |
download | liblss16-b60d205c290e843e05009cc709f2d3d1c1cd4aea.tar.gz liblss16-b60d205c290e843e05009cc709f2d3d1c1cd4aea.tar.bz2 liblss16-b60d205c290e843e05009cc709f2d3d1c1cd4aea.tar.xz |
First commit
Signed-off-by: Mattias Andrée <m@maandree.se>
Diffstat (limited to 'lss16toppm.c')
-rw-r--r-- | lss16toppm.c | 142 |
1 files changed, 142 insertions, 0 deletions
diff --git a/lss16toppm.c b/lss16toppm.c new file mode 100644 index 0000000..db4d448 --- /dev/null +++ b/lss16toppm.c @@ -0,0 +1,142 @@ +/* See LICENSE file for copyright and license details. */ +#include "liblss16.h" +#include <errno.h> +#include <stdio.h> +#include <stdlib.h> +#include <string.h> +#include <unistd.h> + + +static const char *argv0 = "lss16toppm"; + + +static void +writeall(char *buf, size_t n) +{ + ssize_t r; + + while (n) { + r = write(STDOUT_FILENO, buf, n); + if (r < 0) { + if (errno == EINTR) + continue; + fprintf(stderr, "%s: write <stdout>: %s\n", argv0, strerror(errno)); + exit(2); + } + buf += (size_t)r; + n -= (size_t)r; + } +} + + +int +main(int argc, char *argv[]) +{ + char buf[1024], wbuf[1024]; + uint8_t pixels[1024]; + struct liblss16_decoder decoder; + enum liblss16_decode_state state; + ssize_t r; + size_t n, off, consumed, npixels, pending; + enum liblss16_decode_error error; + int head_printed = 0; + int print_colour_map = 0; + int have_stdout, len; + size_t i; + + /* cmdline syntax is inherited from the SYSLINUX implementation, + * it's bad, but it's the way it is, and the way it will remain */ + + if (argc) { + argv0 = *argv++; + argc--; + } + + for (; argc; argc--, argv++) { + if (!strcmp(*argv, "-map")) { + print_colour_map = 1; + } else { + fprintf(stderr, "%s: Unknown option: %s\n", argv0, *argv); + } + } + + liblss16_decoder_init(&decoder); + + pending = 0; + for (;;) { + r = read(STDIN_FILENO, buf, sizeof(buf)); + if (r <= 0) { + if (!r) + break; + if (errno == EINTR) + continue; + fprintf(stderr, "%s: read <stdin>: %s\n", argv0, strerror(errno)); + return 2; + } + n = (size_t)r; + + for (off = 0; off < n;) { + state = liblss16_decode_to_colour_index(&decoder, &buf[off], n - off, &consumed, + pixels, sizeof(pixels) / sizeof(*pixels), &npixels, &error); + off += consumed; + if (state == LIBLSS16_DECODE_ERROR) { + goto error; + } else if (state == LIBLSS16_DECODE_END_OF_IMAGE) { + if (off != n) { + fprintf(stderr, "%s: image file contains extraneous data\n", argv0); + return 1; + } + } + if (!npixels) + continue; + + if (!head_printed) { + head_printed = 1; + len = sprintf(wbuf, "P6\n%u %u\n255\n", decoder.width, decoder.height); + if (len < 0 || (size_t)len >= sizeof(wbuf)) + abort(); + pending = (size_t)len; + if (print_colour_map) { + for (i = 0; i < sizeof(decoder.colour_map) / sizeof(*decoder.colour_map); i++) { + fprintf(stderr, "#%02x%02x%02x=%zu\n", decoder.colour_map[i].r, + decoder.colour_map[i].g, decoder.colour_map[i].b, i); + } + } + } + for (i = 0; i < npixels; i++) { + if (3U > sizeof(wbuf) - pending) { + writeall(wbuf, pending); + pending = 0; + } + wbuf[pending++] = (char)decoder.colour_map[pixels[i]].r; + wbuf[pending++] = (char)decoder.colour_map[pixels[i]].g; + wbuf[pending++] = (char)decoder.colour_map[pixels[i]].b; + } + } + } + if (pending) + writeall(wbuf, pending); + + state = liblss16_decode_to_colour_index(&decoder, NULL, 0, &consumed, pixels, + sizeof(pixels) / sizeof(*pixels), &npixels, &error); + if (state == LIBLSS16_DECODE_ERROR) { + error: + fprintf(stderr, "%s: %s\n", argv0, liblss16_decode_strerror(error)); + return 1; + } else if (state != LIBLSS16_DECODE_END_OF_IMAGE) { + fprintf(stderr, "%s: image file truncated\n", argv0); + return 1; + } + + while (close(STDOUT_FILENO)) { + if (have_stdout && errno == EBADF) + break; + if (errno == EINTR) { + have_stdout = 1; + } else { + fprintf(stderr, "%s: write <stdout>: %s\n", argv0, strerror(errno)); + return 2; + } + } + return 0; +} |