summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--v4l2-apps/util/xc3028-firmware/firmware-tool.c96
1 files changed, 81 insertions, 15 deletions
diff --git a/v4l2-apps/util/xc3028-firmware/firmware-tool.c b/v4l2-apps/util/xc3028-firmware/firmware-tool.c
index eb8e02dff..89b792307 100644
--- a/v4l2-apps/util/xc3028-firmware/firmware-tool.c
+++ b/v4l2-apps/util/xc3028-firmware/firmware-tool.c
@@ -20,11 +20,12 @@
along with this program; if not, write to the Free Software
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
*/
-
#include <sys/stat.h>
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
+#define __USE_GNU
+#include <string.h>
#include <getopt.h>
#include <string.h>
#include <unistd.h>
@@ -586,16 +587,21 @@ struct chunk_hunk {
long pos;
int size;
int need_fix_endian;
+ int hint_method;
struct chunk_hunk *next;
};
-int seek_chunks(struct chunk_hunk *hunk,
+int seek_chunks(struct chunk_hunk *fhunk,
unsigned char *seek, unsigned char *endp, /* File to seek */
unsigned char *fdata, unsigned char *endf) /* Firmware */
{
unsigned char *fpos, *p, *p2, *lastp;
int rc, fsize;
unsigned char *temp_data;
+ struct chunk_hunk *hunk = fhunk;
+ /* Method 3 vars */
+ static unsigned char *base_start = 0;
+ int ini_sig = 8, sig_len = 14, end_sig = 10;
/* Method 1a: Seek for a complete firmware */
for (p = seek; p < endp; p++) {
@@ -610,7 +616,7 @@ int seek_chunks(struct chunk_hunk *hunk,
hunk->size = endf - fdata;
hunk->next = NULL;
hunk->need_fix_endian = 0;
-
+ hunk->hint_method = 0;
return 1;
}
}
@@ -644,13 +650,64 @@ int seek_chunks(struct chunk_hunk *hunk,
hunk->size = endf - fdata;
hunk->next = NULL;
hunk->need_fix_endian = 1;
+ hunk->hint_method = 0;
return 1;
}
}
free(temp_data);
- /* Method 2: Seek for each firmware chunk */
+ /* Method 2: seek for base firmware */
+ if (!base_start)
+ base_start = seek;
+
+ /* Skip if firmware is not a base firmware */
+ if (endf - fdata < 1000)
+ goto method3;
+
+ for (p = base_start; p < endp; p++) {
+ fpos = p;
+ for (p2 = fdata + ini_sig;
+ p2 < fdata + ini_sig + sig_len; p2++,
+ fpos++) {
+ if (*fpos != *p2)
+ break;
+ }
+ if (p2 == fdata + ini_sig + sig_len) {
+ base_start = p - ini_sig;
+
+ p = memmem (base_start, endp-base_start,
+ temp_data + fsize - end_sig, end_sig);
+
+ if (!p) {
+ printf("Method3: found something that looks like a firmware start at %x\n",
+ base_start - seek);
+
+ base_start += ini_sig + sig_len;
+ goto method3;
+ }
+
+ p += end_sig;
+
+ printf("Method3: found firmware at %x, size = %d\n",
+ base_start - seek, p - base_start);
+
+ hunk->data = NULL;
+ hunk->pos = base_start - seek;
+ hunk->size = p - base_start;
+ hunk->next = NULL;
+ hunk->need_fix_endian = 1;
+ hunk->hint_method = 3;
+
+ base_start = p;
+
+ return 2;
+ }
+ }
+
+method3:
+#if 0
+ /* Method 3: Seek for each firmware chunk */
p = seek;
for (p2 = fdata; p2 < endf;) {
int size = *p2 + (*(p2 + 1) << 8);
@@ -662,6 +719,8 @@ int seek_chunks(struct chunk_hunk *hunk,
hunk->pos = -1;
hunk->next = calloc(1, sizeof(hunk));
hunk->need_fix_endian = 0;
+ hunk->hint_method = 0;
+
hunk = hunk->next;
p2 += 2;
@@ -682,8 +741,9 @@ int seek_chunks(struct chunk_hunk *hunk,
hunk->size = size;
hunk->next = calloc(1, sizeof(hunk));
hunk->need_fix_endian = 0;
- hunk = hunk->next;
+ hunk->hint_method = 0;
+ hunk = hunk->next;
break;
}
}
@@ -693,14 +753,14 @@ int seek_chunks(struct chunk_hunk *hunk,
p2 += size;
}
}
-
- return 2;
-
+ return 3;
+#endif
not_found:
+ memset(fhunk, 0, sizeof(struct chunk_hunk));
printf("Couldn't find firmware\n");
-return 0;
+ return 0;
- /* Method 3: Seek for first firmware chunks */
+ /* Method 4: Seek for first firmware chunks */
#if 0
seek_next:
for (p = seek; p < endp; p++) {
@@ -709,9 +769,9 @@ seek_next:
if (*fpos != *p2)
break;
}
- if (p2 > fdata + 2) {
+ if (p2 > fdata + 3) {
int i = 0;
- printf("Found %ld equal bytes at %ld:\n",
+ printf("Found %ld equal bytes at %06x:\n",
p2 - fdata, p - seek);
fpos = p;
lastp = fpos;
@@ -729,7 +789,7 @@ seek_next:
if (fdata == endf) {
printf ("Found all chunks.\n");
- return 1;
+ return 4;
}
}
}
@@ -749,7 +809,7 @@ void seek_firmware(struct firmware *f, char *seek_file, char *write_file) {
FILE *fp;
struct chunk_hunk hunks[f->nr_desc];
- memset (hunks, 0, sizeof(hunks));
+ memset (hunks, 0, sizeof(struct chunk_hunk) * f->nr_desc);
fp=fopen(seek_file, "r");
if (!fp) {
@@ -815,6 +875,12 @@ void seek_firmware(struct firmware *f, char *seek_file, char *write_file) {
for (i = 0; i < f->nr_desc; ++i) {
struct chunk_hunk *hunk = &hunks[i];
+ if (!hunk->size)
+ continue;
+
+ if (hunk->hint_method)
+ fprintf(fp, "\n\t#\n\t# Guessed format ");
+
fprintf(fp, "\n\t#\n\t# Firmware %d, ", i);
list_firmware_desc(fp, &f->desc[i]);
fprintf(fp, "\t#\n\n");
@@ -828,7 +894,7 @@ void seek_firmware(struct firmware *f, char *seek_file, char *write_file) {
f->desc[i].int_freq);
fprintf(fp, "\twrite_le32(%d);\t\t\t# Size\n",
f->desc[i].size);
-
+fflush(fp);
while (hunk) {
if (hunk->data) {
int j;