ref: 313deab55253ee49ef3872491f6805a03b5ea36b
parent: 8f88e04401083d847d8f234645a91067c9c70e1b
author: Rangi <remy.oukaour+rangi42@gmail.com>
date: Wed Sep 1 20:21:10 EDT 2021
Rewrite tool png_dimensions.c, and start using common.h more
--- a/tools/common.h
+++ b/tools/common.h
@@ -1,40 +1,102 @@
#ifndef GUARD_COMMON_H
#define GUARD_COMMON_H
-int __getopt_long_i__;
-#define getopt_long(c, v, s, l) getopt_long(c, v, s, l, &__getopt_long_i__)
+#include <stdio.h>
+#include <stdlib.h>
+#include <stdint.h>
+#include <stdbool.h>
+#include <inttypes.h>
+#include <string.h>
+#include <errno.h>
+#include <unistd.h>
+#include <getopt.h>
-FILE *fopen_verbose(char *filename, char *mode) {
+int getopt_long_index;
+#define getopt_long(c, v, s, l) getopt_long(c, v, s, l, &getopt_long_index)
+
+void *malloc_verbose(size_t size) {
+ void *m = malloc(size);
+ if (!m) {
+ fprintf(stderr, "Could not allocate %zu bytes: %s\n", size, strerror(errno));
+ exit(1);
+ }
+ return m;
+}
+
+FILE *fopen_verbose(const char *filename, char rw) {
+ char mode[3] = {rw, 'b', '\0'};
FILE *f = fopen(filename, mode);
if (!f) {
- fprintf(stderr, "Could not open file: \"%s\"\n", filename);
+ fprintf(stderr, "Could not open file \"%s\": %s\n", filename, strerror(errno));
+ exit(1);
}
return f;
}
-uint8_t *read_u8(char *filename, int *size) {
- FILE *f = fopen_verbose(filename, "rb");
- if (!f) {
+void fread_verbose(uint8_t *data, size_t size, const char *filename, FILE *f) {
+ if (fread(data, 1, size, f) != size) {
+ fprintf(stderr, "Could not read from file \"%s\": %s\n", filename, strerror(errno));
+ fclose(f);
exit(1);
}
- fseek(f, 0, SEEK_END);
- *size = ftell(f);
- rewind(f);
- uint8_t *data = malloc(*size);
- if (*size != (int)fread(data, 1, *size, f)) {
- fprintf(stderr, "Could not read file: \"%s\"\n", filename);
+}
+
+void fwrite_verbose(const uint8_t *data, size_t size, const char *filename, FILE *f) {
+ if (fwrite(data, 1, size, f) != size) {
+ fprintf(stderr, "Could not write to file \"%s\": %s\n", filename, strerror(errno));
+ fclose(f);
exit(1);
}
+}
+
+long file_size(const char *filename, FILE *f) {
+ long size = 0;
+ if (!fseek(f, 0, SEEK_END)) {
+ size = ftell(f);
+ if (size != -1) {
+ rewind(f);
+ }
+ }
+ if (errno) {
+ fprintf(stderr, "Could not measure file \"%s\": %s\n", filename, strerror(errno));
+ exit(1);
+ }
+ return size;
+}
+
+uint8_t *read_u8(const char *filename, long *size) {
+ FILE *f = fopen_verbose(filename, 'r');
+ *size = file_size(filename, f);
+ uint8_t *data = malloc_verbose(*size);
+ fread_verbose(data, *size, filename, f);
fclose(f);
return data;
}
-void write_u8(char *filename, uint8_t *data, int size) {
- FILE *f = fopen_verbose(filename, "wb");
- if (f) {
- fwrite(data, 1, size, f);
+void write_u8(const char *filename, uint8_t *data, size_t size) {
+ FILE *f = fopen_verbose(filename, 'w');
+ fwrite_verbose(data, size, filename, f);
+ fclose(f);
+}
+
+uint32_t read_png_width_verbose(const char *filename) {
+ FILE *f = fopen_verbose(filename, 'r');
+ uint8_t header[16] = {0};
+ fread_verbose(header, sizeof(header), filename, f);
+ static uint8_t expected_header[16] = {
+ 0x89, 'P', 'N', 'G', '\r', '\n', 0x1A, '\n', // signature
+ 0, 0, 0, 13, // IHDR chunk length
+ 'I', 'H', 'D', 'R', // IHDR chunk type
+ };
+ if (memcmp(header, expected_header, sizeof(header))) {
+ fprintf(stderr, "Not a valid PNG file: \"%s\"\n", filename);
fclose(f);
+ exit(1);
}
+ uint8_t bytes[4] = {0};
+ fread_verbose(bytes, sizeof(bytes), filename, f);
+ fclose(f);
+ return (bytes[0] << 24) | (bytes[1] << 16) | (bytes[2] << 8) | bytes[3];
}
#endif // GUARD_COMMON_H
--- a/tools/gfx.c
+++ b/tools/gfx.c
@@ -104,7 +104,7 @@
}
struct Graphic {
- int size;
+ long size;
uint8_t *data;
};
@@ -270,31 +270,6 @@
free(interleaved);
}
-int png_get_width(char *filename) {
- FILE *f = fopen_verbose(filename, "rb");
- if (!f) {
- exit(1);
- }
-
- const int OFFSET_WIDTH = 16;
- uint8_t bytes[4];
- fseek(f, OFFSET_WIDTH, SEEK_SET);
- size_t size = 4;
- size_t result = fread(bytes, 1, size, f);
- fclose(f);
- if (result != size) {
- fprintf(stderr, "Could not read file at offset 0x%x: \"%s\"\n", OFFSET_WIDTH, filename);
- exit(1);
- }
-
- int width = 0;
- for (int i = 0; i < 4; i++) {
- width |= bytes[i] << (8 * (3 - i));
- }
- return width;
-}
-
-
int main(int argc, char *argv[]) {
get_args(argc, argv);
argc -= optind;
@@ -319,7 +294,7 @@
usage();
exit(1);
}
- int width = png_get_width(Options.png_file);
+ int width = read_png_width_verbose(Options.png_file);
interleave(&graphic, width);
}
if (Options.remove_duplicates) {
--- a/tools/png_dimensions.c
+++ b/tools/png_dimensions.c
@@ -1,57 +1,25 @@
-#include <stdio.h>
-#include <stdint.h>
-#include <stdlib.h>
+#include "common.h"
-void usage(void) {
- fprintf(stderr, "Usage: png_dimensions infile outfile\n");
- exit(1);
+void usage() {
+ fprintf(stderr, "Usage: png_dimensions in.png out.dimensions\n");
}
-void output_dimensions(char* png_filename, char* out_filename) {
- FILE* f;
- int width, height;
- int i;
- uint8_t bytes[4];
- uint8_t output;
-
- f = fopen(png_filename, "rb");
- if (f == NULL) {
- fprintf(stderr, "failed to open file %s\n", png_filename);
+uint8_t read_dimensions(const char *filename) {
+ uint32_t width_px = read_png_width_verbose(filename);
+ if (width_px != 40 && width_px != 48 && width_px != 56) {
+ fprintf(stderr, "Not a valid width for \"%s\": %" PRIu32 " px\n", filename, width_px);
exit(1);
}
-
- // width
- fseek(f, 16, SEEK_SET);
- int size = fread(bytes, 1, 4, f);
- fclose(f);
- if (size != 4) {
- fprintf(stderr, "failed to read at offset 0x10 in file %s\n", png_filename);
- exit(1);
- }
-
- width = 0;
- for (i = 0; i < 4; i++) {
- width |= bytes[i] << (8 * (3 - i));
- }
- width >>= 3;
- height = width;
-
- output = width & 0xf;
- output |= (height & 0xf) << 4;
-
- f = fopen(out_filename, "wb");
- if (f == NULL) {
- fprintf(stderr, "failed to open file %s\n", out_filename);
- exit(1);
- }
- fwrite(&output, 1, 1, f);
- fclose(f);
+ uint8_t width_tiles = (uint8_t)(width_px / 8);
+ return (width_tiles << 4) | width_tiles;
}
-int main(int argc, char* argv[]) {
+int main(int argc, char *argv[]) {
if (argc < 3) {
usage();
+ exit(1);
}
- output_dimensions(argv[1], argv[2]);
+ uint8_t output_byte = read_dimensions(argv[1]);
+ write_u8(argv[2], &output_byte, 1);
return 0;
}
--- a/tools/stadium.c
+++ b/tools/stadium.c
@@ -1,10 +1,3 @@
-#include <stdio.h>
-#include <stdlib.h>
-#include <string.h>
-#include <stdint.h>
-#include <stdbool.h>
-#include <getopt.h>
-
#include "common.h"
// The Game Boy cartridge header stores a global checksum at 0x014E-0x014F
@@ -160,7 +153,7 @@
}
char *filename = argv[0];
- int filesize;
+ long filesize;
uint8_t *file = read_u8(filename, &filesize);
calculate_checksums(file, filesize, base);
write_u8(filename, file, filesize);