// Copyright (c) Cesanta Software Limited // All rights reserved. // This program is used to pack arbitrary data into a C binary. It takes // a list of files as an input, and produces a .c data file that contains // contents of all these files as a collection of byte arrays. // // Usage: // 1. Compile this file: // cc -o pack pack.c // // 2. Convert list of files into single .c: // ./pack file1.data file2.data > fs.c // // 3. In your application code, you can access files using this function: // const char *mg_unpack(const char *file_name, size_t *size); // // 4. Build your app with fs.c: // cc -o my_app my_app.c fs.c #include #include #include #include #include static const char *code = "static int scmp(const char *a, const char *b) {\n" " while (*a && (*a == *b)) a++, b++;\n" " return *(const unsigned char *) a - *(const unsigned char *) b;\n" "}\n" "const char *mg_unlist(size_t no) {\n" " return packed_files[no].name;\n" "}\n" "const char *mg_unpack(const char *name, size_t *size, time_t *mtime) {\n" " const struct packed_file *p;\n" " for (p = packed_files; p->name != NULL; p++) {\n" " if (scmp(p->name, name) != 0) continue;\n" " if (size != NULL) *size = p->size - 1;\n" " if (mtime != NULL) *mtime = p->mtime;\n" " return (const char *) p->data;\n" " }\n" " return NULL;\n" "}\n"; int main(int argc, char *argv[]) { int i, j, ch; const char *strip_prefix = ""; printf("%s", "#include \n"); printf("%s", "#include \n"); printf("%s", "#include \n"); printf("%s", "\n"); printf("%s", "#if defined(__cplusplus)\nextern \"C\" {\n#endif\n"); printf("%s", "const char *mg_unlist(size_t no);\n"); printf("%s", "const char *mg_unpack(const char *, size_t *, time_t *);\n"); printf("%s", "#if defined(__cplusplus)\n}\n#endif\n\n"); for (i = 1; i < argc; i++) { if (strcmp(argv[i], "-s") == 0) { strip_prefix = argv[++i]; } else if (strcmp(argv[i], "-h") == 0 || strcmp(argv[i], "--help") == 0) { fprintf(stderr, "Usage: %s[-s STRIP_PREFIX] files...\n", argv[0]); exit(EXIT_FAILURE); } else { char ascii[12]; FILE *fp = fopen(argv[i], "rb"); if (fp == NULL) { fprintf(stderr, "Cannot open [%s]: %s\n", argv[i], strerror(errno)); exit(EXIT_FAILURE); } printf("static const unsigned char v%d[] = {\n", i); for (j = 0; (ch = fgetc(fp)) != EOF; j++) { if (j == (int) sizeof(ascii)) { printf(" // %.*s\n", j, ascii); j = 0; } ascii[j] = (char) ((ch >= ' ' && ch <= '~' && ch != '\\') ? ch : '.'); printf(" %3u,", ch); } // Append zero byte at the end, to make text files appear in memory // as nul-terminated strings. // printf(" 0 // %.*s\n", (int) sizeof(ascii), ascii); printf(" 0 // %.*s\n};\n", j, ascii); fclose(fp); } } printf("%s", "\nstatic const struct packed_file {\n"); printf("%s", " const char *name;\n"); printf("%s", " const unsigned char *data;\n"); printf("%s", " size_t size;\n"); printf("%s", " time_t mtime;\n"); printf("%s", "} packed_files[] = {\n"); for (i = 1; i < argc; i++) { struct stat st; const char *name = argv[i]; size_t n = strlen(strip_prefix); if (strcmp(argv[i], "-s") == 0) { i++; continue; } stat(argv[i], &st); if (strncmp(name, strip_prefix, n) == 0) name += n; printf(" {\"/%s\", v%d, sizeof(v%d), %lu},\n", name, i, i, (unsigned long) st.st_mtime); } printf("%s", " {NULL, NULL, 0, 0}\n"); printf("%s", "};\n\n"); printf("%s", code); return EXIT_SUCCESS; }