aboutsummaryrefslogtreecommitdiffstats
path: root/roms/SLOF/tools/sloffs.c
diff options
context:
space:
mode:
Diffstat (limited to 'roms/SLOF/tools/sloffs.c')
-rw-r--r--roms/SLOF/tools/sloffs.c763
1 files changed, 763 insertions, 0 deletions
diff --git a/roms/SLOF/tools/sloffs.c b/roms/SLOF/tools/sloffs.c
new file mode 100644
index 000000000..264b0eabc
--- /dev/null
+++ b/roms/SLOF/tools/sloffs.c
@@ -0,0 +1,763 @@
+/******************************************************************************
+ * Copyright (c) 2008, 2009 Adrian Reber
+ * All rights reserved.
+ * This program and the accompanying materials
+ * are made available under the terms of the BSD License
+ * which accompanies this distribution, and is available at
+ * http://www.opensource.org/licenses/bsd-license.php
+ *
+ * Contributors:
+ * Adrian Reber - initial implementation
+ *****************************************************************************/
+
+#include <stdint.h>
+#include <stdio.h>
+#include <sys/types.h>
+#include <sys/stat.h>
+#include <unistd.h>
+#include <sys/mman.h>
+#include <fcntl.h>
+#include <string.h>
+#include <stdlib.h>
+#include <byteswap.h>
+#include <getopt.h>
+#include <time.h>
+#include <errno.h>
+
+#include <calculatecrc.h>
+#include <crclib.h>
+
+#define VERSION 1
+
+#ifdef _BIG_ENDIAN
+#define cpu_to_be64(x) (x)
+#define be64_to_cpu(x) (x)
+#define be16_to_cpu(x) (x)
+#define be32_to_cpu(x) (x)
+#else
+#define cpu_to_be64(x) bswap_64(x)
+#define be64_to_cpu(x) bswap_64(x)
+#define be16_to_cpu(x) bswap_16(x)
+#define be32_to_cpu(x) bswap_32(x)
+#endif
+
+
+/* no board dependencies wanted here, let's hardcode SLOF's
+ * magic strings here */
+
+#define FLASHFS_MAGIC "magic123"
+#define FLASHFS_PLATFORM_MAGIC "JS2XBlade"
+#define FLASHFS_PLATFORM_REVISION "1"
+
+/* there seems to be no structure defined anywhere in the code
+ * which resembles the actual sloffs/romfs file header;
+ * so defining it here for now */
+
+struct sloffs {
+ uint64_t next;
+ uint64_t len;
+ uint64_t flags;
+ uint64_t data;
+ char *name;
+};
+
+/* sloffs metadata size:
+ * 4 * 8: 4 * uint64_t + (filename length) */
+#define SLOFFS_META (4 * 8)
+#define ALIGN64(x) (((x) + 7) & ~7)
+
+static struct sloffs *
+next_file_mm(struct sloffs *sloffs)
+{
+ return (struct sloffs *)((unsigned char *)sloffs +
+ be64_to_cpu(sloffs->next));
+}
+
+static int
+next_file(const int fd, struct sloffs *sloffs)
+{
+ int ret;
+ uint64_t size;
+ uint64_t offset;
+ char *name;
+
+ offset = 0;
+
+ /* if sloffs is not all NULL we want the next file
+ * else we just take the first file */
+ if (sloffs->name && sloffs->len && sloffs->data) {
+ offset = be64_to_cpu(sloffs->next);
+ /* we already read over the header; skip it in the seek */
+ offset -= be64_to_cpu(sloffs->data);
+ free(sloffs->name);
+ sloffs->name = NULL;
+ lseek(fd, offset, SEEK_CUR);
+ } else {
+ lseek(fd, offset, SEEK_SET);
+ }
+
+ ret = read(fd, sloffs, SLOFFS_META);
+ if (ret == -1)
+ return -1;
+ /* read the size of the header */
+ size = be64_to_cpu(sloffs->data);
+ /* get the size of the filename */
+ size -= SLOFFS_META;
+ name = malloc(size);
+
+ ret = read(fd, name, size);
+ if (ret == -1) {
+ free(name);
+ return -1;
+ }
+ sloffs->name = name;
+ return 0;
+}
+
+static struct sloffs *
+find_file_mm(const void *data, const char *name)
+{
+ struct sloffs *sloffs = (struct sloffs *)data;
+
+ for (;;) {
+ if (!strcmp((char *)&sloffs->name, name))
+ return sloffs;
+
+ if (be64_to_cpu(sloffs->next) == 0)
+ break;
+ sloffs = next_file_mm(sloffs);
+ }
+ return NULL;
+}
+
+static struct sloffs *
+find_file(const int fd, const char *name, struct sloffs *sloffs)
+{
+ memset(sloffs, 0, sizeof(struct sloffs));
+
+ if (next_file(fd, sloffs))
+ return NULL;
+
+ for (;;) {
+ if (!strcmp(sloffs->name, name))
+ return sloffs;
+
+ if (be64_to_cpu(sloffs->next) == 0)
+ break;
+ if (next_file(fd, sloffs))
+ return NULL;
+ }
+
+ free(sloffs->name);
+ return NULL;
+}
+
+static struct stH *
+sloffs_header_mm(const void *data)
+{
+ struct sloffs *sloffs;
+ struct stH *header;
+
+ /* find the "header" file with all the information about
+ * the flash image */
+ sloffs = find_file_mm(data, "header");
+ if (!sloffs) {
+ printf("sloffs file \"header\" not found. aborting...\n");
+ return NULL;
+ }
+
+ header = (struct stH *)((unsigned char *)sloffs +
+ be64_to_cpu(sloffs->data));
+ return header;
+}
+
+static struct stH *
+sloffs_header(const int fd)
+{
+ struct sloffs file;
+ struct sloffs *sloffs;
+ struct stH *header;
+ ssize_t rc;
+
+ header = (struct stH *)malloc(sizeof(struct stH));
+
+ /* find the "header" file with all the information about
+ * the flash image */
+ sloffs = find_file(fd, "header", &file);
+ if (!sloffs) {
+ printf("sloffs file \"header\" not found. aborting...\n");
+ return NULL;
+ }
+
+ rc = read(fd, header, sizeof(struct stH));
+ if (rc != sizeof(struct stH)) {
+ printf("Reading header, rc %ld, errno %d\n", rc, errno);
+ free(header);
+ header = NULL;
+ }
+ free(sloffs->name);
+ return header;
+}
+
+static uint64_t
+header_length_mm(const void *data)
+{
+ struct sloffs *sloffs;
+
+ /* find the "header" file with all the information about
+ * the flash image */
+ sloffs = find_file_mm(data, "header");
+ if (!sloffs) {
+ printf("sloffs file \"header\" not found. aborting...\n");
+ return 0;
+ }
+ return be64_to_cpu(sloffs->len);
+}
+
+static uint64_t
+header_length(const int fd)
+{
+ struct sloffs file;
+ struct sloffs *sloffs;
+
+ /* find the "header" file with all the information about
+ * the flash image */
+ sloffs = find_file(fd, "header", &file);
+ if (!sloffs) {
+ printf("sloffs file \"header\" not found. aborting...\n");
+ return 0;
+ }
+
+ free(sloffs->name);
+ return be64_to_cpu(sloffs->len);
+}
+
+static void
+update_modification_time(struct stH *header)
+{
+ struct tm *tm;
+ time_t caltime;
+ char dastr[16] = { 0, };
+ uint64_t date;
+
+ /* update modification date
+ * copied from create_crc.c */
+ caltime = time(NULL);
+ tm = localtime(&caltime);
+ strftime(dastr, 15, "0x%Y%m%d%H%M", tm);
+ date = cpu_to_be64(strtoll(dastr, NULL, 16));
+
+ /* this does not match the definition from
+ * struct stH, but we immitate the bug from
+ * flash image creation in create_crc.c.
+ * The date is in mdate and time in padding2. */
+ memcpy(&(header->mdate), &date, 8);
+}
+
+static void
+update_crc(void *data)
+{
+ uint64_t crc;
+ struct stH *header = sloffs_header_mm(data);
+ uint64_t len = be64_to_cpu(header->flashlen);
+
+ /* calculate header CRC */
+ header->ui64CRC = 0;
+ crc = checkCRC(data, header_length_mm(data), 0);
+ header->ui64CRC = cpu_to_be64(crc);
+ /* calculate flash image CRC */
+ crc = checkCRC(data, len, 0);
+ *(uint64_t *)(data + len - 8) = cpu_to_be64(crc);
+}
+
+static uint64_t
+check_image_crc(const int fd, uint64_t len)
+{
+ uint64_t crc;
+ uint64_t i;
+ uint64_t read_bytes;
+ unsigned char buffer[4096];
+
+ lseek(fd, 0, SEEK_SET);
+ crc = 0;
+ read_bytes = 0;
+ while (read_bytes < len) {
+ i = read(fd, buffer, 4096);
+ read_bytes += i;
+ if (read_bytes > len)
+ i -= read_bytes - len;
+ crc = calCRCword(buffer, i, crc);
+ }
+ return crc;
+}
+static void
+sloffs_append(const int file, const char *name, const char *dest)
+{
+ void *append;
+ unsigned char *write_data;
+ void *write_start;
+ int fd;
+ int out;
+ struct stat stat;
+ struct stH *header;
+ uint64_t new_len;
+ struct sloffs *sloffs;
+ struct sloffs new_file;
+ uint64_t read_len;
+ int i;
+ ssize_t rc;
+
+ fd = open(name, O_RDONLY);
+
+ if (fd == -1) {
+ perror(name);
+ exit(1);
+ }
+
+ fstat(fd, &stat);
+ append = mmap(NULL, stat.st_size, PROT_READ, MAP_SHARED, fd, 0);
+ header = sloffs_header(file);
+
+ if (!header)
+ return;
+
+ new_len = ALIGN64(stat.st_size) + be64_to_cpu(header->flashlen);
+ /* add the length of the sloffs file meta information */
+ new_len += SLOFFS_META;
+ /* add the length of the filename */
+ new_len += ALIGN64(strlen(name) + 1);
+
+ out = open(dest, O_CREAT | O_RDWR | O_TRUNC, 00666);
+
+ if (out == -1) {
+ perror(dest);
+ exit(1);
+ }
+
+ /* write byte at the end to be able to mmap it */
+ lseek(out, new_len - 1, SEEK_SET);
+ rc = write(out, "", 1);
+ if (rc != 1) {
+ printf("Extending file failed, rc %ld, errno %d\n", rc, errno);
+ exit(1);
+ }
+ write_start = mmap(NULL, new_len, PROT_READ | PROT_WRITE,
+ MAP_SHARED, out, 0);
+
+ memset(write_start, 0, new_len);
+ memset(&new_file, 0, sizeof(struct sloffs));
+
+ new_file.len = cpu_to_be64(stat.st_size);
+ new_file.data = cpu_to_be64(SLOFFS_META + ALIGN64(strlen(name) + 1));
+
+ if (write_start == MAP_FAILED) {
+ perror("mmap");
+ exit(1);
+ }
+
+ lseek(file, 0, SEEK_SET);
+ write_data = write_start;
+ read_len = be64_to_cpu(header->flashlen);
+ for (;;) {
+ i = read(file, write_data, read_len);
+ if (i < 0) {
+ perror("read");
+ exit(1);
+ }
+ if (i == 0)
+ break;
+ write_data += i;
+ read_len -= i;
+ }
+ /* -8: overwrite old CRC */
+ write_data = write_start + be64_to_cpu(header->flashlen) - 8;
+ memcpy(write_data, &new_file, SLOFFS_META);
+ write_data += SLOFFS_META;
+ /* write the filename */
+ memcpy(write_data, name, strlen(name));
+ write_data += ALIGN64(strlen(name) + 1 );
+ memcpy(write_data, append, stat.st_size);
+
+ write_data = write_start;
+
+ /* find last file */
+ sloffs = (struct sloffs *)write_start;
+ for (;;) {
+ if (be64_to_cpu(sloffs->next) == 0)
+ break;
+ sloffs = next_file_mm(sloffs);
+ }
+ /* get the distance to the next file */
+ sloffs->next = ALIGN64(be64_to_cpu(sloffs->len));
+ /* and the offset were the data starts */
+ sloffs->next += be64_to_cpu(sloffs->data);
+ /* and we have to skip the end of file marker
+ * if one is there; if the last uint64_t is -1
+ * it is an end of file marker; this is a bit dangerous
+ * but there is no other way to detect the end of
+ * file marker */
+ if ((uint64_t)be64_to_cpu(*(uint64_t *)((unsigned char *)sloffs +
+ sloffs->next)) == (uint64_t)-1ULL)
+ sloffs->next += 8;
+
+ sloffs->next = cpu_to_be64(sloffs->next);
+
+ free(header);
+ /* update new length of flash image */
+ header = sloffs_header_mm(write_start);
+ header->flashlen = cpu_to_be64(new_len);
+
+ update_modification_time(header);
+
+ update_crc(write_start);
+
+ munmap(append, stat.st_size);
+ munmap(write_start, new_len);
+ close(fd);
+ close(out);
+}
+
+static void print_header_date(void *dptr)
+{
+ uint8_t *date = dptr;
+
+ if (date[2] || date[3] || date[4] || date[5] || date[6] || date[7]) {
+ printf("%02x%02x-%02x-%02x %02x:%02x", date[2], date[3],
+ date[4], date[5], date[6], date[7]);
+ } else {
+ printf("N/A");
+ }
+
+}
+
+static void
+sloffs_dump(const int fd)
+{
+ void *data;
+ struct stH *header;
+ struct sloffs file;
+ int i;
+ uint64_t crc;
+ uint64_t header_len;
+ ssize_t rc;
+
+ header = sloffs_header(fd);
+
+ if (!header)
+ return;
+
+ if (memcmp(FLASHFS_MAGIC, header->magic, strlen(FLASHFS_MAGIC))) {
+ printf("sloffs magic not found. "
+ "probably not a valid SLOF flash image. aborting...\n");
+ return;
+ }
+ printf(" Magic : %s\n", header->magic);
+ printf(" Platform : %s\n", header->platform_name);
+ printf(" Version : %s\n", header->version);
+ /* there is a bug in the date position;
+ * it should be at header->date, but it is at (header->date + 2) */
+ printf(" Build Date : ");
+ print_header_date(header->date);
+ printf("\n");
+ printf(" Modify Date : ");
+ print_header_date(header->mdate);
+ printf("\n");
+ printf(" Image Length: %ld", be64_to_cpu(header->flashlen));
+ printf(" (0x%lx) bytes\n", be64_to_cpu(header->flashlen));
+ printf(" Revision : %s\n", header->platform_revision);
+ crc = be64_to_cpu(header->ui64CRC);
+ printf(" Header CRC : 0x%016lx CRC check: ", crc);
+ /* to test the CRC of the header we need to know the actual
+ * size of the file and not just the size of the data
+ * which could be easily obtained with sizeof(struct stH);
+ * the actual size can only be obtained from the filesystem
+ * meta information */
+ header_len = header_length(fd);
+ /* no copy the header to memory to crc test it */
+ data = malloc(header_len);
+ lseek(fd, 0, SEEK_SET);
+ rc = read(fd, data, header_len);
+ if (rc != (ssize_t) header_len) {
+ printf("Reading header failed, rc %zd, errno %d\n", rc, errno);
+ return;
+ }
+ crc = calCRCword((unsigned char *)data, header_length(fd), 0);
+ free(data);
+ if (!crc)
+ printf("[OK]");
+ else
+ printf("[FAILED]");
+ printf("\n");
+
+ crc = be64_to_cpu(header->flashlen);
+ /* move to the CRC */
+ lseek(fd, crc - 8, SEEK_SET);
+ /* read it */
+ rc = read(fd, &crc, 8);
+ if (rc != 8) {
+ printf("Reading crc failed, rc %zd, errno %d\n", rc, errno);
+ return;
+ }
+ crc = be64_to_cpu(crc);
+ printf(" Image CRC : 0x%016lx CRC check: ", crc);
+ crc = check_image_crc(fd, be64_to_cpu(header->flashlen));
+ if (!crc)
+ printf("[OK]");
+ else
+ printf("[FAILED]");
+ printf("\n");
+
+ /* count number of files */
+ i = 0;
+ memset(&file, 0, sizeof(struct sloffs));
+ if (next_file(fd, &file))
+ return;
+ for (;;) {
+ i++;
+
+ if (be64_to_cpu(file.next) == 0)
+ break;
+ if (next_file(fd, &file))
+ return;
+ }
+ free(file.name);
+ printf(" Files : %d\n", i);
+ free(header);
+}
+
+static void
+sloffs_list(const int fd)
+{
+ const char *name_header = "File Name";
+ unsigned int i;
+ unsigned int max;
+ unsigned int line;
+ struct sloffs file;
+ uint64_t offset = 0;
+
+ memset(&file, 0, sizeof(struct sloffs));
+
+ if (next_file(fd, &file))
+ return;
+
+ /* find largest name */
+ max = strlen(name_header);
+ for (;;) {
+ if (max < strlen((char *)file.name))
+ max = strlen((char *)file.name);
+
+ if (be64_to_cpu(file.next) == 0)
+ break;
+ if (next_file(fd, &file))
+ return;
+ }
+
+ free(file.name);
+
+
+ /* have at least two spaces between name and size column */
+ max += 2;
+
+ /* header for listing */
+ line = printf(" Offset ");
+ line += printf("%s", name_header);
+ for (i = 0; i < max - strlen(name_header); i++)
+ line += printf(" ");
+ line += printf("Size ");
+ line += printf("Flags\n");
+ printf(" ");
+ for (i = 0; i <= line; i++)
+ printf("=");
+ printf("\n");
+
+ memset(&file, 0, sizeof(struct sloffs));
+
+ if (next_file(fd, &file))
+ return;
+
+ for (;;) {
+ printf(" 0x%08lx", offset);
+ offset += be64_to_cpu(file.next);
+ printf(" %s", file.name);
+ for (i = 0; i < max - strlen(file.name); i++)
+ printf(" ");
+
+ printf("%07ld ", be64_to_cpu(file.len));
+ printf("(0x%06lx)", be64_to_cpu(file.len));
+ printf(" 0x%08lx\n", be64_to_cpu(file.flags));
+
+ if (be64_to_cpu(file.next) == 0)
+ break;
+ if (next_file(fd, &file))
+ return;
+ }
+ free(file.name);
+}
+
+static void
+sloffs_copy(const int file, const char *name)
+{
+ uint64_t len;
+ int out;
+ unsigned char *write_buf;
+ int i;
+ struct stH *header;
+ ssize_t rc;
+
+ header = sloffs_header(file);
+
+ if (!header)
+ return;
+
+ len = be64_to_cpu(header->flashlen);
+ free(header);
+
+ out = open(name, O_CREAT | O_RDWR | O_TRUNC, 00666);
+
+ if (out == -1) {
+ perror(name);
+ exit(1);
+ }
+ /* write byte at the end to be able to mmap it */
+ lseek(out, len - 1, SEEK_SET);
+ rc = write(out, "", 1);
+ if (rc != 1) {
+ printf("Extending file failed, rc %zd, errno %d\n", rc, errno);
+ exit(1);
+ }
+ write_buf = mmap(NULL, len, PROT_WRITE, MAP_SHARED, out, 0);
+
+ if (write_buf == MAP_FAILED) {
+ perror("mmap");
+ exit(1);
+ }
+
+ lseek(file, 0, SEEK_SET);
+
+ for (;;) {
+ i = read(file, write_buf, len);
+ if (i < 0) {
+ perror("read");
+ exit(1);
+ }
+ if (i == 0)
+ break;
+ write_buf += i;
+ len -= i;
+ }
+
+ munmap(write_buf, len);
+ close(out);
+}
+
+static void
+usage(void)
+{
+ printf("sloffs lists or changes a SLOF flash image\n\n");
+ printf("Usage:\n");
+ printf(" sloffs [OPTION]... [FILE]\n\n");
+ printf("Options:\n");
+ printf(" -h, --help show this help, then exit\n");
+ printf(" -l, --list list all files in the flash image\n");
+ printf(" -v, --version print the version, then exit\n");
+ printf(" -d, --dump dump the information from the header\n");
+ printf(" -a, --append=FILENAME append file at the end of\n");
+ printf(" the existing image\n");
+ printf(" -o, --output=FILENAME if appending a file this parameter\n");
+ printf(" is necessary to specify the name of\n");
+ printf(" the output file\n");
+ printf(" -c, --copy=FILENAME copy SLOF image to specified file\n");
+ printf(" this is especially useful if the\n");
+ printf(" source file is /dev/slof_flash\n");
+ printf("\n");
+ exit(1);
+}
+
+int
+main(int argc, char *argv[])
+{
+ int fd;
+ const struct option loption[] = {
+ { "help", 0, NULL, 'h' },
+ { "list", 0, NULL, 'l' },
+ { "version", 0, NULL, 'v' },
+ { "dump", 0, NULL, 'd' },
+ { "append", 1, NULL, 'a' },
+ { "output", 1, NULL, 'o' },
+ { "copy", 1, NULL, 'o' },
+ { 0, 0, 0, 0 }
+ };
+ const char *soption = "dhlva:o:c:";
+ int c;
+ char mode = 0;
+ char *append = NULL;
+ char *output = NULL;
+
+ for (;;) {
+ c = getopt_long(argc, argv, soption, loption, NULL);
+ if (c == -1)
+ break;
+ switch (c) {
+ case 'l':
+ mode = 'l';
+ break;
+ case 'v':
+ printf("sloffs (version %d)\n", VERSION);
+ exit(0);
+ case 'd':
+ mode = 'd';
+ break;
+ case 'a':
+ mode = 'a';
+ append = strdup(optarg);
+ break;
+ case 'o':
+ output = strdup(optarg);
+ break;
+ case 'c':
+ mode = 'c';
+ output = strdup(optarg);
+ break;
+ case 'h':
+ default:
+ usage();
+ }
+ }
+
+ if (optind >= argc)
+ usage();
+
+ fd = open(argv[optind], O_RDONLY);
+
+ if (fd == -1) {
+ perror(argv[optind]);
+ exit(1);
+ }
+
+ lseek(fd, 0, SEEK_SET);
+
+ switch (mode) {
+ case 'l':
+ sloffs_list(fd);
+ break;
+ case 'd':
+ sloffs_dump(fd);
+ break;
+ case 'a':
+ if (!output) {
+ printf("sloffs requires -o, --output=FILENAME"
+ " when in append mode\n\n");
+ usage();
+ }
+ sloffs_append(fd, append, output);
+ break;
+ case 'c':
+ sloffs_copy(fd, output);
+ break;
+ }
+
+ free(append);
+ free(output);
+ close(fd);
+ return 0;
+}