You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
179 lines
4.8 KiB
179 lines
4.8 KiB
/* |
|
* Simple tool to dump the AP_Param contents from an EEPROM dump |
|
* Andrew Tridgell February 2012 |
|
*/ |
|
#include <stdio.h> |
|
#include <stdlib.h> |
|
#include <stdint.h> |
|
|
|
uint8_t eeprom[0x1000]; |
|
|
|
struct EEPROM_header { |
|
uint8_t magic[2]; |
|
uint8_t revision; |
|
uint8_t spare; |
|
}; |
|
|
|
static const uint16_t k_EEPROM_magic0 = 0x50; |
|
static const uint16_t k_EEPROM_magic1 = 0x41; |
|
static const uint16_t k_EEPROM_revision = 6; |
|
|
|
enum ap_var_type { |
|
AP_PARAM_NONE = 0, |
|
AP_PARAM_INT8, |
|
AP_PARAM_INT16, |
|
AP_PARAM_INT32, |
|
AP_PARAM_FLOAT, |
|
AP_PARAM_VECTOR3F, |
|
AP_PARAM_VECTOR6F, |
|
AP_PARAM_MATRIX3F, |
|
AP_PARAM_GROUP |
|
}; |
|
|
|
static const char *type_names[8] = { |
|
"NONE", "INT8", "INT16", "INT32", "FLOAT", "VECTOR3F", "MATRIX6F", "GROUP" |
|
}; |
|
|
|
struct Param_header { |
|
uint32_t key : 8; |
|
uint32_t type : 6; |
|
uint32_t group_element : 18; |
|
}; |
|
|
|
|
|
static const uint8_t _sentinal_key = 0xFF; |
|
static const uint8_t _sentinal_type = 0xFF; |
|
static const uint8_t _sentinal_group = 0xFF; |
|
|
|
static uint8_t type_size(enum ap_var_type type) |
|
{ |
|
switch (type) { |
|
case AP_PARAM_NONE: |
|
case AP_PARAM_GROUP: |
|
return 0; |
|
case AP_PARAM_INT8: |
|
return 1; |
|
case AP_PARAM_INT16: |
|
return 2; |
|
case AP_PARAM_INT32: |
|
return 4; |
|
case AP_PARAM_FLOAT: |
|
return 4; |
|
case AP_PARAM_VECTOR3F: |
|
return 3*4; |
|
case AP_PARAM_VECTOR6F: |
|
return 6*4; |
|
case AP_PARAM_MATRIX3F: |
|
return 3*3*4; |
|
} |
|
printf("unknown type %u\n", (unsigned int)type); |
|
return 0; |
|
} |
|
|
|
static void |
|
fail(const char *why) |
|
{ |
|
fprintf(stderr, "ERROR: %s\n", why); |
|
exit(1); |
|
} |
|
|
|
int |
|
main(int argc, char *argv[]) |
|
{ |
|
FILE *fp; |
|
struct EEPROM_header *header; |
|
struct Param_header *var; |
|
unsigned index; |
|
unsigned i; |
|
|
|
if (argc != 2) { |
|
fail("missing EEPROM file name"); |
|
} |
|
if (NULL == (fp = fopen(argv[1], "rb"))) { |
|
fail("can't open EEPROM file"); |
|
} |
|
if (1 != fread(eeprom, sizeof(eeprom), 1, fp)) { |
|
fail("can't read EEPROM file"); |
|
} |
|
fclose(fp); |
|
|
|
header = (struct EEPROM_header *)&eeprom[0]; |
|
if (header->magic[0] != k_EEPROM_magic0 || |
|
header->magic[1] != k_EEPROM_magic1) { |
|
fail("bad magic in EEPROM file"); |
|
} |
|
if (header->revision != k_EEPROM_revision) { |
|
fail("unsupported EEPROM format revision"); |
|
} |
|
printf("Header OK\n"); |
|
|
|
index = sizeof(*header); |
|
for (;; ) { |
|
uint8_t size; |
|
var = (struct Param_header *)&eeprom[index]; |
|
if (var->key == _sentinal_key || |
|
var->group_element == _sentinal_group || |
|
var->type == _sentinal_type) { |
|
printf("end sentinel at %u\n", index); |
|
break; |
|
} |
|
size = type_size(var->type); |
|
printf("%04x: type %u (%s) key %u group_element %u size %d value ", |
|
index, var->type, type_names[var->type], var->key, var->group_element, size); |
|
index += sizeof(*var); |
|
switch (var->type) { |
|
case AP_PARAM_INT8: |
|
printf("%d\n", (int)*(int8_t *)&eeprom[index]); |
|
break; |
|
case AP_PARAM_INT16: |
|
printf("%d\n", (int)*(int16_t *)&eeprom[index]); |
|
break; |
|
case AP_PARAM_INT32: |
|
printf("%d\n", (int)*(int32_t *)&eeprom[index]); |
|
break; |
|
case AP_PARAM_FLOAT: |
|
printf("%f\n", *(float *)&eeprom[index]); |
|
break; |
|
case AP_PARAM_VECTOR3F: |
|
printf("%f %f %f\n", |
|
*(float *)&eeprom[index], |
|
*(float *)&eeprom[index+4], |
|
*(float *)&eeprom[index+8]); |
|
break; |
|
case AP_PARAM_VECTOR6F: |
|
printf("%f %f %f %f %f %f\n", |
|
*(float *)&eeprom[index], |
|
*(float *)&eeprom[index+4], |
|
*(float *)&eeprom[index+8], |
|
*(float *)&eeprom[index+12], |
|
*(float *)&eeprom[index+16], |
|
*(float *)&eeprom[index+20]); |
|
break; |
|
case AP_PARAM_MATRIX3F: |
|
printf("%f %f %f %f %f %f %f %f %f\n", |
|
*(float *)&eeprom[index], |
|
*(float *)&eeprom[index+4], |
|
*(float *)&eeprom[index+8], |
|
*(float *)&eeprom[index+12], |
|
*(float *)&eeprom[index+16], |
|
*(float *)&eeprom[index+20], |
|
*(float *)&eeprom[index+24], |
|
*(float *)&eeprom[index+28], |
|
*(float *)&eeprom[index+32]); |
|
break; |
|
default: |
|
printf("NONE\n"); |
|
break; |
|
} |
|
for (i = 0; i < size; i++) { |
|
printf(" %02x", eeprom[index + i]); |
|
} |
|
printf("\n"); |
|
index += size; |
|
if (index >= sizeof(eeprom)) { |
|
fflush(stdout); |
|
fail("missing end sentinel"); |
|
} |
|
} |
|
return 0; |
|
}
|
|
|