#include <assert.h>
+#include <ctype.h>
#include <errno.h>
#include <stdbool.h>
+#include <stddef.h>
#include <stdio.h>
#include <stdint.h>
#include <stdlib.h>
.vendor = "",
.support_list =
"SupportList:\r\n"
- "{product_name:Talon AD7200,product_ver:1.0.0,special_id:00000000}\r\n",
+ "{product_name:AD7200,product_ver:1.0.0,special_id:00000000}\r\n",
.part_trail = 0x00,
.soft_ver = NULL,
.first_sysupgrade_partition = "os-image",
.last_sysupgrade_partition = "file-system",
},
+ /** Firmware layout for the Archer A6 v3 */
+ {
+ .id = "ARCHER-A6-V3",
+ .vendor = "",
+ .support_list =
+ "SupportList:\n"
+ "{product_name:Archer A6,product_ver:3.0.0,special_id:55530000}\n"
+ "{product_name:Archer A6,product_ver:3.0.0,special_id:54570000}\n",
+ .part_trail = 0x00,
+ .soft_ver = "soft_ver:1.0.5\n",
+ .partitions = {
+ {"fs-uboot", 0x00000, 0x40000},
+ {"firmware", 0x40000, 0xf60000},
+ {"default-mac", 0xfa0000, 0x00200},
+ {"pin", 0xfa0200, 0x00100},
+ {"device-id", 0xfa0300, 0x00100},
+ {"product-info", 0xfa0400, 0x0fc00},
+ {"default-config", 0xfb0000, 0x08000},
+ {"ap-def-config", 0xfb8000, 0x08000},
+ {"user-config", 0xfc0000, 0x0a000},
+ {"ag-config", 0xfca000, 0x04000},
+ {"certificate", 0xfce000, 0x02000},
+ {"ap-config", 0xfd0000, 0x06000},
+ {"router-config", 0xfd6000, 0x06000},
+ {"favicon", 0xfdc000, 0x02000},
+ {"logo", 0xfde000, 0x02000},
+ {"partition-table", 0xfe0000, 0x00800},
+ {"soft-version", 0xfe0800, 0x00100},
+ {"support-list", 0xfe0900, 0x00200},
+ {"profile", 0xfe0b00, 0x03000},
+ {"extra-para", 0xfe3b00, 0x00100},
+ {"radio", 0xff0000, 0x10000},
+ {NULL, 0, 0}
+ },
+ .first_sysupgrade_partition = "os-image",
+ .last_sysupgrade_partition = "file-system",
+ },
/** Firmware layout for the C60v1 */
{
.id = "ARCHER-C60-V1",
"{product_name:Archer C7,product_ver:5.0.0,special_id:4B520000}\n",
.part_trail = 0x00,
- .soft_ver = "soft_ver:1.0.0\n",
+ .soft_ver = "soft_ver:7.0.0\n",
/* We're using a dynamic kernel/rootfs split here */
.partitions = {
.last_sysupgrade_partition = "file-system"
},
+ /** Firmware layout for the EAP235-Wall v1 */
+ {
+ .id = "EAP235-WALL-V1",
+ .support_list =
+ "SupportList:\r\n"
+ "EAP235-Wall(TP-Link|UN|AC1200-D):1.0\r\n",
+ .part_trail = PART_TRAIL_NONE,
+ .soft_ver = NULL,
+ .soft_ver_compat_level = 1,
+
+ .partitions = {
+ {"fs-uboot", 0x00000, 0x80000},
+ {"partition-table", 0x80000, 0x02000},
+ {"default-mac", 0x90000, 0x01000},
+ {"support-list", 0x91000, 0x00100},
+ {"product-info", 0x91100, 0x00400},
+ {"soft-version", 0x92000, 0x00100},
+ {"firmware", 0xa0000, 0xd20000},
+ {"user-config", 0xdc0000, 0x30000},
+ {"mutil-log", 0xf30000, 0x80000},
+ {"oops", 0xfb0000, 0x40000},
+ {"radio", 0xff0000, 0x10000},
+ {NULL, 0, 0}
+ },
+
+ .first_sysupgrade_partition = "os-image",
+ .last_sysupgrade_partition = "file-system"
+ },
+
/** Firmware layout for the EAP245 v1 */
{
.id = "EAP245-V1",
parts[4] = read_file("file-system", rootfs_image, add_jffs2_eof, file_system_partition);
/* Some devices need the extra-para partition to accept the firmware */
- if (strcasecmp(info->id, "ARCHER-A7-V5") == 0 ||
+ if (strcasecmp(info->id, "ARCHER-A6-V3") == 0 ||
+ strcasecmp(info->id, "ARCHER-A7-V5") == 0 ||
strcasecmp(info->id, "ARCHER-C2-V3") == 0 ||
strcasecmp(info->id, "ARCHER-C7-V4") == 0 ||
strcasecmp(info->id, "ARCHER-C7-V5") == 0 ||
"Options:\n"
" -h show this help\n"
"\n"
+ "Info about an image:\n"
+ " -i <file> input file to read from\n"
"Create a new image:\n"
" -B <board> create image for the board specified with <board>\n"
" -k <file> read kernel image from the file <file>\n"
return entries;
}
- error(1, 0, "%s", error_msg);
+ if (error_msg) {
+ error(1, 0, "%s", error_msg);
+ }
+
return NULL;
}
+static int firmware_info(const char *input)
+{
+ struct flash_partition_entry pointers[MAX_PARTITIONS] = { };
+ struct flash_partition_entry *e;
+ FILE *fp;
+ int i;
+
+ fp = fopen(input, "r");
+
+ if (read_partition_table(fp, 0x1014, pointers, MAX_PARTITIONS, 0)) {
+ error(1, 0, "Error can not read the partition table (fwup-ptn)");
+ }
+
+ printf("Firmware image partitions:\n");
+ printf("%-8s %-8s %s\n", "base", "size", "name");
+ for (i = 0; i < MAX_PARTITIONS; i++) {
+ e = &pointers[i];
+
+ if (!e->name && !e->base && !e->size)
+ continue;
+
+ printf("%08x %08x %s\n", e->base, e->size, e->name ? e->name : "");
+ }
+
+ e = find_partition(pointers, MAX_PARTITIONS, "soft-version", NULL);
+ if (e) {
+ size_t data_len = e->size - sizeof(struct meta_header);
+ char *buf = malloc(data_len);
+ struct soft_version *s;
+ bool isstr;
+ int i;
+
+ if (!buf)
+ error(1, errno, "Failed to alloc buffer");
+
+ if (fseek(fp, 0x1014 + e->base + sizeof(struct meta_header), SEEK_SET))
+ error(1, errno, "Can not seek in the firmware");
+
+ if (fread(buf, data_len, 1, fp) != 1)
+ error(1, errno, "Can not read fwup-ptn data from the firmware");
+
+ /* Check for string ignoring padding character */
+ isstr = true;
+ for (i = 0; i < data_len - 1; i++) {
+ if (!isascii(buf[i])) {
+ isstr = false;
+ break;
+ }
+ }
+
+ printf("\n[Software version]\n");
+ if (isstr) {
+ fwrite(buf, data_len, 1, stdout);
+ putchar('\n');
+ } else if (data_len >= offsetof(struct soft_version, rev)) {
+ s = (struct soft_version *)buf;
+
+ printf("Version: %d.%d.%d\n", s->version_major, s->version_minor, s->version_patch);
+ printf("Date: %02x%02x-%02x-%02x\n", s->year_hi, s->year_lo, s->month, s->day);
+ } else {
+ printf("Failed to parse data\n");
+ }
+
+ free(buf);
+ }
+
+ e = find_partition(pointers, MAX_PARTITIONS, "support-list", NULL);
+ if (e) {
+ char buf[128];
+ size_t length;
+ size_t bytes;
+
+ if (fseek(fp, 0x1014 + e->base + sizeof(struct meta_header), SEEK_SET))
+ error(1, errno, "Can not seek in the firmware");
+
+ printf("\n[Support list]\n");
+ for (length = e->size - sizeof(struct meta_header); length; length -= bytes) {
+ bytes = fread(buf, 1, length > sizeof(buf) ? sizeof(buf) : length, fp);
+ if (bytes <= 0)
+ error(1, errno, "Can not read fwup-ptn data from the firmware");
+
+ puts(buf);
+ }
+ }
+
+ e = find_partition(pointers, MAX_PARTITIONS, "partition-table", NULL);
+ if (e) {
+ struct flash_partition_entry parts[MAX_PARTITIONS] = { };
+
+ if (read_partition_table(fp, 0x1014 + e->base + 4, parts, MAX_PARTITIONS, 1)) {
+ error(1, 0, "Error can not read the partition table (partition)");
+ }
+
+ printf("\n[Partition table]\n");
+ printf("%-8s %-8s %s\n", "base", "size", "name");
+ for (i = 0; i < MAX_PARTITIONS; i++) {
+ e = &parts[i];
+
+ if (!e->name && !e->base && !e->size)
+ continue;
+
+ printf("%08x %08x %s\n", e->base, e->size, e->name ? e->name : "");
+ }
+ }
+
+ fclose(fp);
+
+ return 0;
+}
+
static void write_ff(FILE *output_file, size_t size)
{
char buf[4096];
}
int main(int argc, char *argv[]) {
- const char *board = NULL, *kernel_image = NULL, *rootfs_image = NULL, *output = NULL;
+ const char *info_image = NULL, *board = NULL, *kernel_image = NULL, *rootfs_image = NULL, *output = NULL;
const char *extract_image = NULL, *output_directory = NULL, *convert_image = NULL;
bool add_jffs2_eof = false, sysupgrade = false;
unsigned rev = 0;
while (true) {
int c;
- c = getopt(argc, argv, "B:k:r:o:V:jSh:x:d:z:");
+ c = getopt(argc, argv, "i:B:k:r:o:V:jSh:x:d:z:");
if (c == -1)
break;
switch (c) {
+ case 'i':
+ info_image = optarg;
+ break;
+
case 'B':
board = optarg;
break;
}
}
- if (extract_image || output_directory) {
+ if (info_image) {
+ firmware_info(info_image);
+ } else if (extract_image || output_directory) {
if (!extract_image)
error(1, 0, "No factory/oem image given via -x <file>. Output directory is only valid with -x");
if (!output_directory)