QModem/application/qfirehose/src/firehose_protocol.c
2025-03-06 18:21:10 +08:00

2298 lines
84 KiB
C
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/*
Copyright 2023 Quectel Wireless Solutions Co.,Ltd
Quectel hereby grants customers of Quectel a license to use, modify,
distribute and publish the Software in binary form provided that
customers shall have no right to reverse engineer, reverse assemble,
decompile or reduce to source code form any portion of the Software.
Under no circumstances may customers modify, demonstrate, use, deliver
or disclose any portion of the Software in source code form.
*/
#include "usb_linux.h"
#include <poll.h>
#include <pthread.h>
#include <sys/socket.h>
/*
#define error_return() \
do \
{ \
dbg_time("%s %s %d fail\n", __FILE__, __func__, __LINE__); \
return __LINE__; \
} while (0)
*/
int recv_sc600y_configure_num = 1;
extern const char *q_device_type;
static int fh_recv_cmd_sk[2];
extern unsigned q_module_packet_sign;
extern unsigned q_erase_all_before_download;
extern int update_transfer_bytes(long long bytes_cur);
extern int show_progress();
char file_name_image[128] = {0};
char file_name_image_dir[256] = {0};
typedef struct sparse_header
{
uint32_t magic; /* 0xed26ff3a */
uint16_t major_version; /* (0x1) - reject images with higher major versions */
uint16_t minor_version; /* (0x0) - allow images with higer minor versions */
uint16_t file_hdr_sz; /* 28 bytes for first revision of the file format */
uint16_t chunk_hdr_sz; /* 12 bytes for first revision of the file format */
uint32_t blk_sz; /* block size in bytes, must be a multiple of 4 (4096) */
uint32_t total_blks; /* total blocks in the non-sparse output image */
uint32_t total_chunks; /* total chunks in the sparse input image */
uint32_t image_checksum; /* CRC32 checksum of the original data, counting
"don't care" */
/* as 0. Standard 802.3 polynomial, use a Public Domain */
/* table implementation */
} sparse_header_t;
#define SPARSE_HEADER_MAGIC 0xed26ff3a
typedef struct chunk_header
{
uint16_t chunk_type; /* 0xCAC1 -> raw; 0xCAC2 -> fill; 0xCAC3 -> don't care */
uint16_t reserved1;
uint32_t chunk_sz; /* in blocks in output image */
uint32_t total_sz; /* in bytes of chunk input file including chunk header and
data */
} chunk_header_t;
typedef struct chunk_polymerization_params
{
uint32_t total_chunk_sz;
uint32_t total_sz;
uint16_t total_chunk_count;
// uint16_t file_sector_offset;
} chunk_polymerization_param;
typedef struct SparseImgParams
{
chunk_polymerization_param chunk_polymerization_data[100];
chunk_polymerization_param chunk_polymerization_cac3[100];
uint16_t total_count;
uint16_t total_cac3_count;
uint16_t file_first_sector_offset; //<2F><>һ<EFBFBD><D2BB><EFBFBD><EFBFBD>ͷ<EFBFBD><CDB7>ȡ<EFBFBD><C8A1><EFBFBD><EFBFBD><EFBFBD><EFBFBD><E6B6BC><EFBFBD>Լ<EFBFBD><D4BC><EFBFBD><EFBFBD><EFBFBD><EFBFBD>
} SparseImgParam;
SparseImgParam SparseImgData;
struct fh_configure_cmd
{
const char *type;
const char *MemoryName;
uint32_t Verbose;
uint32_t AlwaysValidate;
uint32_t MaxDigestTableSizeInBytes;
uint32_t MaxPayloadSizeToTargetInBytes;
uint32_t MaxPayloadSizeFromTargetInBytes; // 2048
uint32_t MaxPayloadSizeToTargetInByteSupported; // 16k
uint32_t ZlpAwareHost;
uint32_t SkipStorageInit;
};
struct fh_erase_cmd
{
const char *type;
// uint32_t PAGES_PER_BLOCK;
uint32_t SECTOR_SIZE_IN_BYTES;
// char label[32];
uint32_t last_sector;
uint32_t num_partition_sectors;
// uint32_t physical_partition_number;
uint32_t start_sector;
};
struct fh_program_cmd
{
const char *type;
char *filename;
char *sparse;
uint32_t filesz;
// uint32_t PAGES_PER_BLOCK;
uint32_t SECTOR_SIZE_IN_BYTES;
// char label[32];
// uint32_t last_sector;
uint32_t num_partition_sectors;
uint32_t physical_partition_number;
uint32_t start_sector;
uint32_t file_sector_offset;
uint32_t UNSPARSE_FILE_SIZE;
// char sparse[16];
};
struct fh_response_cmd
{
const char *type;
const char *value;
uint32_t rawmode;
uint32_t MaxPayloadSizeToTargetInBytes;
};
struct fh_log_cmd
{
const char *type;
};
struct fh_patch_cmd
{
const char *type;
char *filename;
uint32_t filesz;
uint32_t SECTOR_SIZE_IN_BYTES;
uint32_t num_partition_sectors;
};
struct fh_cmd_header
{
const char *type;
};
struct fh_vendor_defines
{
const char *type; // "vendor"
};
struct fh_cmd
{
union
{
struct fh_cmd_header cmd;
struct fh_configure_cmd cfg;
struct fh_erase_cmd erase;
struct fh_program_cmd program;
struct fh_response_cmd response;
struct fh_log_cmd log;
struct fh_patch_cmd patch;
struct fh_vendor_defines vdef;
};
int part_upgrade;
char xml_original_data[512];
};
#define fh_cmd_num 1024 // AG525 have more than 64 partition
struct fh_data
{
const char *firehose_dir;
const void *usb_handle;
unsigned MaxPayloadSizeToTargetInBytes;
unsigned fh_cmd_count;
unsigned fh_patch_count;
unsigned ZlpAwareHost;
struct fh_cmd fh_cmd_table[fh_cmd_num];
unsigned xml_tx_size;
unsigned xml_rx_size;
char xml_tx_buf[1024];
char xml_rx_buf[1024];
};
static const char *fh_xml_find_value(const char *xml_line, const char *key, char **ppend)
{
char *pchar = strstr(xml_line, key);
char *pend;
if (!pchar)
{
if (strcmp(key, "sparse")) dbg_time("%s: no key %s in %s\n", __func__, key, xml_line);
return NULL;
}
pchar += strlen(key);
if (pchar[0] != '=' && pchar[1] != '"')
{
dbg_time("%s: no start %s in %s\n", __func__, "=\"", xml_line);
return NULL;
}
pchar += strlen("=\"");
pend = strstr(pchar, "\"");
if (!pend)
{
dbg_time("%s: no end %s in %s\n", __func__, "\"", xml_line);
return NULL;
}
*ppend = pend;
return pchar;
}
static const char *fh_xml_get_value(const char *xml_line, const char *key)
{
static char value[64];
char *pend;
const char *pchar = fh_xml_find_value(xml_line, key, &pend);
if (!pchar)
{
return NULL;
}
int len = pend - pchar;
if (len >= 64) return NULL;
strncpy(value, pchar, pend - pchar);
value[pend - pchar] = '\0';
return value;
}
static void fh_xml_set_value(char *xml_line, const char *key, unsigned value)
{
char *pend;
const char *pchar = fh_xml_find_value(xml_line, key, &pend);
char value_str[32];
char *tmp_line = malloc(strlen(xml_line) + 1 + sizeof(value_str));
if (!pchar || !tmp_line)
{
if (tmp_line)
{
free(tmp_line);
tmp_line = NULL;
}
return;
}
strcpy(tmp_line, xml_line);
snprintf(value_str, sizeof(value_str), "%u", value);
tmp_line[pchar - xml_line] = '\0';
strcat(tmp_line, value_str);
strcat(tmp_line, pend);
strcpy(xml_line, tmp_line);
free(tmp_line);
}
static int fh_parse_xml_line(const char *xml_line, struct fh_cmd *fh_cmd)
{
const char *pchar = NULL;
size_t len = strlen(xml_line);
memset(fh_cmd, 0, sizeof(struct fh_cmd));
strncpy(fh_cmd->xml_original_data, xml_line, 512);
if (fh_cmd->xml_original_data[len - 1] == '\n') fh_cmd->xml_original_data[len - 1] = '\0';
if (strstr(xml_line, "vendor=\"quectel\""))
{
fh_cmd->vdef.type = "vendor";
return 0;
}
else if (!strncmp(xml_line, "<erase ", strlen("<erase ")))
{
fh_cmd->erase.type = "erase";
if (strstr(xml_line, "last_sector"))
{
if ((pchar = fh_xml_get_value(xml_line, "last_sector"))) fh_cmd->erase.last_sector = atoi(pchar);
}
if ((pchar = fh_xml_get_value(xml_line, "start_sector"))) fh_cmd->erase.start_sector = atoi(pchar);
if ((pchar = fh_xml_get_value(xml_line, "num_partition_sectors"))) fh_cmd->erase.num_partition_sectors = atoi(pchar);
if ((pchar = fh_xml_get_value(xml_line, "SECTOR_SIZE_IN_BYTES"))) fh_cmd->erase.SECTOR_SIZE_IN_BYTES = atoi(pchar);
return 0;
}
else if (!strncmp(xml_line, "<program ", strlen("<program ")))
{
fh_cmd->program.type = "program";
if ((pchar = fh_xml_get_value(xml_line, "filename")))
{
fh_cmd->program.filename = strdup(pchar);
if (fh_cmd->program.filename[0] == '\0')
{ // some fw version have blank program line, ignore it.
return -1;
}
}
if ((pchar = fh_xml_get_value(xml_line, "sparse")))
{
fh_cmd->program.sparse = strdup(pchar);
}
else
fh_cmd->program.sparse = NULL;
if ((pchar = fh_xml_get_value(xml_line, "start_sector"))) fh_cmd->program.start_sector = atoi(pchar);
if ((pchar = fh_xml_get_value(xml_line, "num_partition_sectors"))) fh_cmd->program.num_partition_sectors = atoi(pchar);
if ((pchar = fh_xml_get_value(xml_line, "SECTOR_SIZE_IN_BYTES"))) fh_cmd->program.SECTOR_SIZE_IN_BYTES = atoi(pchar);
if (fh_cmd->program.sparse != NULL && !strncasecmp(fh_cmd->program.sparse, "true", 4))
{
if ((pchar = fh_xml_get_value(xml_line, "file_sector_offset"))) fh_cmd->program.file_sector_offset = atoi(pchar);
if ((pchar = fh_xml_get_value(xml_line, "physical_partition_number"))) fh_cmd->program.physical_partition_number = atoi(pchar);
}
return 0;
}
else if (!strncmp(xml_line, "<patch ", strlen("<patch ")))
{
fh_cmd->patch.type = "patch";
pchar = fh_xml_get_value(xml_line, "filename");
if (pchar && strcmp(pchar, "DISK")) return -1;
return 0;
}
else if (!strncmp(xml_line, "<response ", strlen("<response ")))
{
fh_cmd->response.type = "response";
pchar = fh_xml_get_value(xml_line, "value");
if (pchar)
{
if (!strcmp(pchar, "ACK"))
fh_cmd->response.value = "ACK";
else if (!strcmp(pchar, "NAK"))
fh_cmd->response.value = "NAK";
else
fh_cmd->response.value = "OTHER";
}
if (strstr(xml_line, "rawmode"))
{
pchar = fh_xml_get_value(xml_line, "rawmode");
if (pchar)
{
fh_cmd->response.rawmode = !strcmp(pchar, "true");
}
}
else if (strstr(xml_line, "MaxPayloadSizeToTargetInBytes"))
{
pchar = fh_xml_get_value(xml_line, "MaxPayloadSizeToTargetInBytes");
if (pchar)
{
fh_cmd->response.MaxPayloadSizeToTargetInBytes = atoi(pchar);
}
}
return 0;
}
else if (!strncmp(xml_line, "<log ", strlen("<log ")))
{
fh_cmd->program.type = "log";
return 0;
}
error_return();
}
static int fh_parse_xml_file(struct fh_data *fh_data, const char *xml_file)
{
FILE *fp = fopen(xml_file, "rb");
if (fp == NULL)
{
dbg_time("%s fail to fopen(%s), errno: %d (%s)\n", __func__, xml_file, errno, strerror(errno));
error_return();
}
while (fgets(fh_data->xml_tx_buf, fh_data->xml_tx_size, fp))
{
char *xml_line = strstr(fh_data->xml_tx_buf, "<");
char *c_start = NULL;
if (!xml_line) continue;
c_start = strstr(xml_line, "<!--");
if (c_start)
{
char *c_end = strstr(c_start, "-->");
if (c_end)
{
/*
<erase case 1 /> <!-- xxx -->
<!-- xxx --> <erase case 2 />
<!-- <erase case 3 /> -->
*/
char *tmp = strstr(xml_line, "/>");
if (tmp && (tmp < c_start || tmp > c_end))
{
memset(c_start, ' ', c_end - c_start + strlen("-->"));
goto __fh_parse_xml_line;
}
continue;
}
else
{
/*
<!-- line1
<! -- line2 -->
-->
*/
do
{
if (fgets(fh_data->xml_tx_buf, fh_data->xml_tx_size, fp) == NULL)
{
break;
};
xml_line = fh_data->xml_tx_buf;
} while (!strstr(xml_line, "-->") && strstr(xml_line, "<!--"));
continue;
}
}
__fh_parse_xml_line:
if (xml_line)
{
char *tag = NULL;
tag = strstr(xml_line, "<erase ");
if (!tag)
{
tag = strstr(xml_line, "<program ");
if (!tag)
{
tag = strstr(xml_line, "<patch ");
}
}
if (tag)
{
if (!fh_parse_xml_line(tag, &fh_data->fh_cmd_table[fh_data->fh_cmd_count]))
{
fh_data->fh_cmd_count++;
if (strstr(tag, "<patch ")) fh_data->fh_patch_count++;
if (fh_data->fh_cmd_count >= fh_cmd_num)
{
dbg_time("too many fh_cmd, you need to increase fh_cmd_num\n");
exit(-1);
}
}
}
else if (!strstr(xml_line, "<?xml") && !strcmp(xml_line, "<data>") && !strcmp(xml_line, "</data>") && !strcmp(xml_line, "<patches>") && !strcmp(xml_line, "<patches>"))
{
dbg_time("unspport xml_line '%s'\n", xml_line);
exit(-1);
}
}
}
fclose(fp);
return 0;
}
static int fh_fixup_program_cmd(struct fh_data *fh_data, struct fh_cmd *fh_cmd, long *filesize_out)
{
char full_path[512] = {0};
char unix_filename_tmp[256] = {0};
char *ptmp;
FILE *fp;
long filesize = 0;
uint32_t num_partition_sectors = fh_cmd->program.num_partition_sectors;
int image_in_firehose_dir = 0;
char *unix_filename = strdup(fh_cmd->program.filename);
if (unix_filename == NULL)
{
error_return();
}
while ((ptmp = strchr(unix_filename, '\\')))
{
*ptmp = '/';
}
if (is_upgrade_fimeware_zip_7z)
{
int i;
char *p2 = strrchr(unix_filename, '/');
if (p2 == NULL)
{
memmove(unix_filename_tmp, unix_filename, strlen(unix_filename));
image_in_firehose_dir = 1;
}
else
{
memmove(unix_filename_tmp, p2 + 1, strlen(p2) - 1);
}
memset(file_name_image, 0, sizeof(file_name_image));
memset(file_name_image_dir, 0, sizeof(file_name_image_dir));
for (i = 0; i < file_name_b.file_name_count; i++)
{
if (strstr(file_name_b.file_backup_c[i].zip_file_name_backup, unix_filename_tmp))
{
if (image_in_firehose_dir)
{
if (strstr(file_name_b.file_backup_c[i].zip_file_dir_backup, "update/firehose"))
{
memmove(file_name_image, file_name_b.file_backup_c[i].zip_file_name_backup, strlen(file_name_b.file_backup_c[i].zip_file_name_backup));
memmove(file_name_image_dir, file_name_b.file_backup_c[i].zip_file_dir_backup, strlen(file_name_b.file_backup_c[i].zip_file_dir_backup));
break;
}
}
else
{
memmove(file_name_image, file_name_b.file_backup_c[i].zip_file_name_backup, strlen(file_name_b.file_backup_c[i].zip_file_name_backup));
memmove(file_name_image_dir, file_name_b.file_backup_c[i].zip_file_dir_backup, strlen(file_name_b.file_backup_c[i].zip_file_dir_backup));
break;
}
}
}
if (file_name_image[0] != '\0')
{
memset(zip_cmd_buf, 0, sizeof(zip_cmd_buf));
if (is_upgrade_fimeware_only_zip)
{
snprintf(zip_cmd_buf, sizeof(zip_cmd_buf), "unzip -o -q %.240s '*%.200s' -d /tmp/ > %s", fh_data->firehose_dir, file_name_image_dir, ZIP_PROCESS_INFO);
}
else
{
snprintf(zip_cmd_buf, sizeof(zip_cmd_buf), "7z x %.240s -o/tmp/ %.200s > %s", fh_data->firehose_dir, file_name_image_dir, ZIP_PROCESS_INFO);
}
dbg_time("%s zip_cmd_buf:%s\n", __func__, zip_cmd_buf);
if (-1 == system(zip_cmd_buf))
{
dbg_time("%s system return error\n", __func__);
return -1;
}
usleep(1000);
snprintf(full_path, sizeof(full_path), "/tmp/%.240s", file_name_image_dir);
dbg_time("%s full_path:%s\n", __func__, full_path);
}
}
else
{
snprintf(full_path, sizeof(full_path), "%.255s/%.240s", fh_data->firehose_dir, unix_filename);
}
if (access(full_path, R_OK))
{
fh_cmd->program.num_partition_sectors = 0;
dbg_time("fail to access %s, errno: %d (%s)\n", full_path, errno, strerror(errno));
if (unix_filename)
{
free(unix_filename);
unix_filename = NULL;
}
error_return();
}
fp = fopen(full_path, "rb");
if (!fp)
{
fh_cmd->program.num_partition_sectors = 0;
dbg_time("fail to fopen %s, errno: %d (%s)\n", full_path, errno, strerror(errno));
if (unix_filename)
{
free(unix_filename);
unix_filename = NULL;
}
error_return();
}
fseek(fp, 0, SEEK_END);
filesize = ftell(fp);
*filesize_out = filesize;
fclose(fp);
if (filesize <= 0)
{
dbg_time("fail to ftell %s, errno: %d (%s)\n", full_path, errno, strerror(errno));
fh_cmd->program.num_partition_sectors = 0;
fh_cmd->program.filesz = 0;
if (unix_filename)
{
free(unix_filename);
unix_filename = NULL;
}
error_return();
}
fh_cmd->program.filesz = filesize;
fh_cmd->program.num_partition_sectors = filesize / fh_cmd->program.SECTOR_SIZE_IN_BYTES;
if (filesize % fh_cmd->program.SECTOR_SIZE_IN_BYTES) fh_cmd->program.num_partition_sectors += 1;
if (!strncasecmp(unix_filename, "gpt_empty0.bin", 14))
{
fh_cmd->program.num_partition_sectors -= 1;
}
if (num_partition_sectors != fh_cmd->program.num_partition_sectors)
{
fh_xml_set_value(fh_cmd->xml_original_data, "num_partition_sectors", fh_cmd->program.num_partition_sectors);
}
if (is_upgrade_fimeware_zip_7z)
{
unlink(full_path);
}
free(unix_filename);
return 0;
}
static int _fh_recv_cmd(struct fh_data *fh_data, struct fh_cmd *fh_cmd, unsigned timeout)
{
int ret;
char *xml_line;
char *pend;
memset(fh_cmd, 0, sizeof(struct fh_cmd));
ret = qusb_noblock_read(fh_data->usb_handle, fh_data->xml_rx_buf, fh_data->xml_rx_size, 1, timeout);
if (ret <= 0)
{
return -1;
}
fh_data->xml_rx_buf[ret] = '\0';
xml_line = fh_data->xml_rx_buf;
while (*xml_line)
{
xml_line = strstr(xml_line, "<?xml version=");
if (xml_line == NULL)
{
if (fh_cmd->cmd.type == 0)
{
dbg_time("{{{%s}}}", fh_data->xml_rx_buf);
error_return();
}
else
{
break;
}
}
xml_line += strlen("<?xml version=");
xml_line = strstr(xml_line, "<data>");
if (xml_line == NULL)
{
dbg_time("{{{%s}}}", fh_data->xml_rx_buf);
error_return();
}
xml_line += strlen("<data>");
if (xml_line[0] == '\n') xml_line++;
if (!strncmp(xml_line, "<response ", strlen("<response ")))
{
fh_parse_xml_line(xml_line, fh_cmd);
pend = strstr(xml_line, "/>");
pend += 2;
dbg_time("%.*s\n", (int)(pend - xml_line), xml_line);
xml_line = pend + 1;
}
else if (!strncmp(xml_line, "<log ", strlen("<log ")))
{
if (fh_cmd->cmd.type && strcmp(fh_cmd->cmd.type, "log"))
{
dbg_time("{{{%s}}}", fh_data->xml_rx_buf);
break;
}
fh_parse_xml_line(xml_line, fh_cmd);
pend = strstr(xml_line, "/>");
pend += 2;
{
char *prn = xml_line;
while (prn < pend)
{
if (*prn == '\r' || *prn == '\n') *prn = '.';
prn++;
}
}
dbg_time("%.*s\n", (int)(pend - xml_line), xml_line);
xml_line = pend + 1;
}
else
{
dbg_time("unkonw %s", xml_line);
error_return();
}
}
if (fh_cmd->cmd.type) return 0;
error_return();
}
static void *fh_recv_cmd_thread(void *arg)
{
struct fh_data *fh_data = (struct fh_data *)arg;
struct fh_cmd fh_rx_cmd;
while (_fh_recv_cmd(fh_data, &fh_rx_cmd, -1) == 0)
{
if (strncmp(fh_rx_cmd.cmd.type, "log", strlen("log")))
{
if (write(fh_recv_cmd_sk[1], &fh_rx_cmd, sizeof(fh_rx_cmd)) == -1)
{
};
}
}
return NULL;
}
static int fh_recv_cmd(struct fh_data *fh_data, struct fh_cmd *fh_cmd, unsigned timeout, int ignore_timeout)
{
struct pollfd pollfds[] = {{fh_recv_cmd_sk[0], POLLIN, 0}};
int ret = poll(pollfds, 1, timeout);
(void)fh_data;
if (ret == 1 && (pollfds[0].revents & POLLIN))
{
ret = read(fh_recv_cmd_sk[0], fh_cmd, sizeof(struct fh_cmd));
if (ret == sizeof(struct fh_cmd)) return 0;
}
else if (ret == 0 && ignore_timeout)
{
return __LINE__;
}
error_return();
}
static int fh_wait_response_cmd(struct fh_data *fh_data, struct fh_cmd *fh_cmd, unsigned timeout)
{
while (1)
{
int ret = fh_recv_cmd(fh_data, fh_cmd, timeout, 0);
if (ret != 0) error_return();
if (strstr(fh_cmd->cmd.type, "log")) continue;
return 0;
}
error_return();
}
static int fh_send_cmd(struct fh_data *fh_data, const struct fh_cmd *fh_cmd)
{
int tx_len = 0;
char *pstart, *pend;
char *xml_buf = fh_data->xml_tx_buf;
unsigned xml_size = fh_data->xml_tx_size;
xml_buf[0] = '\0';
snprintf(xml_buf + strlen(xml_buf), xml_size, "<?xml version=\"1.0\" encoding=\"UTF-8\" ?>\n");
snprintf(xml_buf + strlen(xml_buf), xml_size, "<data>\n");
pstart = xml_buf + strlen(xml_buf);
if (!strcmp(fh_cmd->cmd.type, "vendor"))
{
snprintf(xml_buf + strlen(xml_buf), xml_size, "%s", fh_cmd->xml_original_data);
}
else if (!strcmp(fh_cmd->cmd.type, "erase"))
{
snprintf(xml_buf + strlen(xml_buf), xml_size, "%s", fh_cmd->xml_original_data);
}
else if (!strcmp(fh_cmd->cmd.type, "program"))
{
if (fh_cmd->program.sparse != NULL && !strncasecmp(fh_cmd->program.sparse, "true", 4))
{
snprintf(xml_buf + strlen(xml_buf), xml_size,
"<program filename=\"%.120s\" SECTOR_SIZE_IN_BYTES=\"%d\" "
"num_partition_sectors=\"%d\" physical_partition_number=\"%d\" start_sector=\"%d\" "
"file_sector_offset=\"%d\" sparse=\"%.120s\" UNSPARSE_FILE_SIZE=\"%d\" />",
fh_cmd->program.filename, fh_cmd->program.SECTOR_SIZE_IN_BYTES, fh_cmd->program.num_partition_sectors, fh_cmd->program.physical_partition_number,
fh_cmd->program.start_sector, fh_cmd->program.file_sector_offset, fh_cmd->program.sparse, fh_cmd->program.UNSPARSE_FILE_SIZE);
}
else
snprintf(xml_buf + strlen(xml_buf), xml_size, "%s", fh_cmd->xml_original_data);
}
else if (!strcmp(fh_cmd->cmd.type, "patch"))
{
snprintf(xml_buf + strlen(xml_buf), xml_size, "%s", fh_cmd->xml_original_data);
}
else if (!strcmp(fh_cmd->cmd.type, "configure"))
{
snprintf(xml_buf + strlen(xml_buf), xml_size,
"<configure MemoryName=\"%.8s\" Verbose=\"%d\" AlwaysValidate=\"%d\" "
"MaxDigestTableSizeInBytes=\"%d\" MaxPayloadSizeToTargetInBytes=\"%d\" "
"ZlpAwareHost=\"%d\" SkipStorageInit=\"%d\" />",
fh_cmd->cfg.MemoryName, fh_cmd->cfg.Verbose, fh_cmd->cfg.AlwaysValidate, fh_cmd->cfg.MaxDigestTableSizeInBytes, fh_cmd->cfg.MaxPayloadSizeToTargetInBytes,
fh_cmd->cfg.ZlpAwareHost, fh_cmd->cfg.SkipStorageInit);
}
else if (!strcmp(fh_cmd->cmd.type, "setbootablestoragedrive"))
{
snprintf(xml_buf + strlen(xml_buf), xml_size, "<setbootablestoragedrive value=\"%d\" />", !strcmp(q_device_type, "ufs") ? 1 : 0);
}
else if (!strcmp(fh_cmd->cmd.type, "reset"))
{
snprintf(xml_buf + strlen(xml_buf), xml_size, "<power DelayInSeconds=\"%u\" value=\"reset\" />", 10);
}
else
{
dbg_time("%s unkonw fh_cmd->cmd.type=%s\n", __func__, fh_cmd->cmd.type);
error_return();
}
pend = xml_buf + strlen(xml_buf);
dbg_time("%.*s\n", (int)(pend - pstart), pstart);
// snprintf(xml_buf + strlen(xml_buf), xml_size, "\n</data>");
if (!strcmp(fh_cmd->cmd.type, "setbootablestoragedrive") || !strcmp(fh_cmd->cmd.type, "reset") || !strcmp(fh_cmd->cmd.type, "configure"))
{
snprintf(xml_buf + strlen(xml_buf), xml_size, "\n</data>\n");
}
else
snprintf(xml_buf + strlen(xml_buf), xml_size, "\n</data>");
tx_len = qusb_noblock_write(fh_data->usb_handle, xml_buf, strlen(xml_buf), strlen(xml_buf), 3000, fh_data->ZlpAwareHost);
if ((size_t)tx_len == strlen(xml_buf)) return 0;
error_return();
}
static int fh_send_cfg_cmd(struct fh_data *fh_data, const char *device_type)
{
struct fh_cmd fh_cfg_cmd;
struct fh_cmd fh_rx_cmd;
memset(&fh_cfg_cmd, 0x00, sizeof(fh_cfg_cmd));
fh_cfg_cmd.cfg.type = "configure";
fh_cfg_cmd.cfg.MemoryName = device_type;
fh_cfg_cmd.cfg.Verbose = 0;
fh_cfg_cmd.cfg.AlwaysValidate = 0;
fh_cfg_cmd.cfg.SkipStorageInit = 0;
fh_cfg_cmd.cfg.ZlpAwareHost = fh_data->ZlpAwareHost; // only sdx20 support zlp set to 0 by 20180822
if (!strcmp(device_type, "emmc") || !strcmp(device_type, "ufs"))
{
fh_cfg_cmd.cfg.MaxDigestTableSizeInBytes = 8192;
fh_cfg_cmd.cfg.MaxPayloadSizeToTargetInBytes = 1048576;
fh_cfg_cmd.cfg.MaxPayloadSizeFromTargetInBytes = 8192;
fh_cfg_cmd.cfg.MaxPayloadSizeToTargetInByteSupported = 1048576;
}
else
{
fh_cfg_cmd.cfg.MaxDigestTableSizeInBytes = 2048;
fh_cfg_cmd.cfg.MaxPayloadSizeToTargetInBytes = 8192;
fh_cfg_cmd.cfg.MaxPayloadSizeFromTargetInBytes = 2048;
fh_cfg_cmd.cfg.MaxPayloadSizeToTargetInByteSupported = 8192;
}
fh_send_cmd(fh_data, &fh_cfg_cmd);
if (fh_wait_response_cmd(fh_data, &fh_rx_cmd, 3000) != 0) error_return();
if (!strcmp(fh_rx_cmd.response.value, "NAK") && fh_rx_cmd.response.MaxPayloadSizeToTargetInBytes)
{
fh_cfg_cmd.cfg.MaxPayloadSizeToTargetInBytes = fh_rx_cmd.response.MaxPayloadSizeToTargetInBytes;
fh_cfg_cmd.cfg.MaxPayloadSizeToTargetInByteSupported = fh_rx_cmd.response.MaxPayloadSizeToTargetInBytes;
fh_send_cmd(fh_data, &fh_cfg_cmd);
if (fh_wait_response_cmd(fh_data, &fh_rx_cmd, 3000) != 0) error_return();
}
if (strcmp(fh_rx_cmd.response.value, "ACK") != 0) error_return();
fh_data->MaxPayloadSizeToTargetInBytes = fh_cfg_cmd.cfg.MaxPayloadSizeToTargetInBytes;
return 0;
}
static int fh_send_setbootablestoragedrive_cmd(struct fh_data *fh_data)
{
struct fh_cmd fh_0_cmd;
fh_0_cmd.cmd.type = "setbootablestoragedrive";
return fh_send_cmd(fh_data, &fh_0_cmd);
}
static int fh_send_reset_cmd(struct fh_data *fh_data)
{
struct fh_cmd fh_reset_cmd;
fh_reset_cmd.cmd.type = "reset";
return fh_send_cmd(fh_data, &fh_reset_cmd);
}
static int fh_send_rawmode_image(struct fh_data *fh_data, const struct fh_cmd *fh_cmd, unsigned timeout)
{
char full_path[512] = {0};
char unix_filename_tmp[256] = {0};
char read_chunk_header_buf[64] = {0};
char *ptmp;
FILE *fp;
size_t filesize, filesend;
int image_in_firehose_dir = 0;
char *unix_filename = strdup(fh_cmd->program.filename);
if (unix_filename == NULL)
{
error_return();
}
void *pbuf = malloc(fh_data->MaxPayloadSizeToTargetInBytes);
if (pbuf == NULL)
{
if (unix_filename)
{
free(unix_filename);
unix_filename = NULL;
}
error_return();
}
while ((ptmp = strchr(unix_filename, '\\')))
{
*ptmp = '/';
}
if (is_upgrade_fimeware_zip_7z)
{
int i;
char *p2 = strrchr(unix_filename, '/');
if (p2 == NULL)
{
memmove(unix_filename_tmp, unix_filename, strlen(unix_filename));
image_in_firehose_dir = 1;
}
else
{
memmove(unix_filename_tmp, p2 + 1, strlen(p2) - 1);
}
memset(file_name_image, 0, sizeof(file_name_image));
memset(file_name_image_dir, 0, sizeof(file_name_image_dir));
for (i = 0; i < file_name_b.file_name_count; i++)
{
if (strstr(file_name_b.file_backup_c[i].zip_file_name_backup, unix_filename_tmp))
{
if (image_in_firehose_dir)
{
if (strstr(file_name_b.file_backup_c[i].zip_file_dir_backup, "update/firehose"))
{
memmove(file_name_image, file_name_b.file_backup_c[i].zip_file_name_backup, strlen(file_name_b.file_backup_c[i].zip_file_name_backup));
memmove(file_name_image_dir, file_name_b.file_backup_c[i].zip_file_dir_backup, strlen(file_name_b.file_backup_c[i].zip_file_dir_backup));
break;
}
}
else
{
memmove(file_name_image, file_name_b.file_backup_c[i].zip_file_name_backup, strlen(file_name_b.file_backup_c[i].zip_file_name_backup));
memmove(file_name_image_dir, file_name_b.file_backup_c[i].zip_file_dir_backup, strlen(file_name_b.file_backup_c[i].zip_file_dir_backup));
break;
}
}
}
if (file_name_image[0] != '\0')
{
memset(zip_cmd_buf, 0, sizeof(zip_cmd_buf));
if (is_upgrade_fimeware_only_zip)
{
snprintf(zip_cmd_buf, sizeof(zip_cmd_buf), "unzip -o -q %.240s '*%.200s' -d /tmp/ > %s", fh_data->firehose_dir, file_name_image_dir, ZIP_PROCESS_INFO);
}
else
{
snprintf(zip_cmd_buf, sizeof(zip_cmd_buf), "7z x %.240s -o/tmp/ %.200s > %s", fh_data->firehose_dir, file_name_image_dir, ZIP_PROCESS_INFO);
}
dbg_time("%s zip_cmd_buf:%s\n", __func__, zip_cmd_buf);
if (-1 == system(zip_cmd_buf))
{
dbg_time("%s system return error\n", __func__);
return -1;
}
usleep(1000);
snprintf(full_path, sizeof(full_path), "/tmp/%.240s", file_name_image_dir);
dbg_time("%s full_path:%s\n", __func__, full_path);
}
}
else
{
snprintf(full_path, sizeof(full_path), "%.255s/%.240s", fh_data->firehose_dir, unix_filename);
}
fp = fopen(full_path, "rb");
if (!fp)
{
dbg_time("fail to fopen %s, errno: %d (%s)\n", full_path, errno, strerror(errno));
if (unix_filename)
{
free(unix_filename);
unix_filename = NULL;
}
if (pbuf)
{
free(pbuf);
pbuf = NULL;
}
error_return();
}
if (fh_cmd->program.sparse != NULL && !strncasecmp(fh_cmd->program.sparse, "true", 4))
{
filesize = fh_cmd->program.UNSPARSE_FILE_SIZE;
filesend = 0;
fseek(fp, fh_cmd->program.file_sector_offset, SEEK_SET);
}
else
{
fseek(fp, 0, SEEK_END);
filesize = ftell(fp);
filesend = 0;
fseek(fp, 0, SEEK_SET);
}
dbg_time("send %s, filesize=%zd\n", unix_filename, filesize);
if (!strncasecmp(unix_filename, "gpt_empty0.bin", 14))
{
filesize -= 512;
}
int idx = -1;
if (fh_cmd->program.sparse != NULL && !strncasecmp(fh_cmd->program.sparse, "true", 4))
{
size_t reads = 0;
while (1)
{
chunk_header_t *chunk_header;
size_t read_header = fread(read_chunk_header_buf, 1, sizeof(chunk_header_t), fp);
if (read_header <= 0)
{
dbg_time("%s fread failed\n", __func__);
}
chunk_header = (chunk_header_t *)read_chunk_header_buf;
#if 0
printf("chunk_header->chunk_type = %0x\n", chunk_header->chunk_type);
printf("chunk_header->reserved1 = %0x\n", chunk_header->reserved1);
printf("chunk_header->chunk_sz = %d\n", chunk_header->chunk_sz);
printf("chunk_header->total_sz = %d\n", chunk_header->total_sz);
#endif
uint32_t chunk_data_sz = 0;
chunk_data_sz = (chunk_header->total_sz - 0xC);
update_transfer_bytes(chunk_data_sz);
if (!((++idx) % 0x80))
{
printf(".");
fflush(stdout);
}
while (chunk_data_sz >= fh_data->MaxPayloadSizeToTargetInBytes)
{
reads = fread(pbuf, 1, fh_data->MaxPayloadSizeToTargetInBytes, fp);
if (reads > 0)
{
if (reads % fh_cmd->program.SECTOR_SIZE_IN_BYTES)
{
memset((uint8_t *)pbuf + reads, 0, fh_cmd->program.SECTOR_SIZE_IN_BYTES - (reads % fh_cmd->program.SECTOR_SIZE_IN_BYTES));
reads += fh_cmd->program.SECTOR_SIZE_IN_BYTES - (reads % fh_cmd->program.SECTOR_SIZE_IN_BYTES);
}
size_t writes = qusb_noblock_write(fh_data->usb_handle, pbuf, reads, reads, timeout, fh_data->ZlpAwareHost);
if (reads != writes)
{
dbg_time("%s send fail reads=%zd, writes=%zd\n", __func__, reads, writes);
dbg_time("%s send fail filesend=%zd, filesize=%zd\n", __func__, filesend, filesize);
break;
}
filesend += reads;
// dbg_time("filesend=%zd, filesize=%zd\n", filesend, filesize);
}
else
{
break;
}
chunk_data_sz -= fh_data->MaxPayloadSizeToTargetInBytes;
}
if (chunk_data_sz > 0)
{
reads = fread(pbuf, 1, chunk_data_sz, fp);
if (reads > 0)
{
if (reads % fh_cmd->program.SECTOR_SIZE_IN_BYTES)
{
memset((uint8_t *)pbuf + reads, 0, fh_cmd->program.SECTOR_SIZE_IN_BYTES - (reads % fh_cmd->program.SECTOR_SIZE_IN_BYTES));
reads += fh_cmd->program.SECTOR_SIZE_IN_BYTES - (reads % fh_cmd->program.SECTOR_SIZE_IN_BYTES);
}
size_t writes = qusb_noblock_write(fh_data->usb_handle, pbuf, reads, reads, timeout, fh_data->ZlpAwareHost);
if (reads != writes)
{
dbg_time("%s send fail reads=%zd, writes=%zd\n", __func__, reads, writes);
dbg_time("%s send fail filesend=%zd, filesize=%zd\n", __func__, filesend, filesize);
break;
}
filesend += reads;
// dbg_time("filesend=%zd, filesize=%zd\n", filesend, filesize);
}
else
{
break;
}
}
if (filesend >= filesize)
{
dbg_time("%s filesend=%zd, filesize=%zd\n", __func__, filesend, filesize);
break;
}
}
}
else
{
while (filesend < filesize)
{
size_t reads;
// printf("fh_data->MaxPayloadSizeToTargetInBytes:%d\n",
// fh_data->MaxPayloadSizeToTargetInBytes);
if (filesize < (filesend + fh_data->MaxPayloadSizeToTargetInBytes))
{
reads = fread(pbuf, 1, filesize - filesend, fp);
}
else
reads = fread(pbuf, 1, fh_data->MaxPayloadSizeToTargetInBytes, fp);
update_transfer_bytes(reads);
if (!((++idx) % 0x80))
{
printf(".");
fflush(stdout);
}
if (reads > 0)
{
if (reads % fh_cmd->program.SECTOR_SIZE_IN_BYTES)
{
memset((uint8_t *)pbuf + reads, 0, fh_cmd->program.SECTOR_SIZE_IN_BYTES - (reads % fh_cmd->program.SECTOR_SIZE_IN_BYTES));
reads += fh_cmd->program.SECTOR_SIZE_IN_BYTES - (reads % fh_cmd->program.SECTOR_SIZE_IN_BYTES);
}
size_t writes = qusb_noblock_write(fh_data->usb_handle, pbuf, reads, reads, timeout, fh_data->ZlpAwareHost);
if (reads != writes)
{
dbg_time("%s send fail reads=%zd, writes=%zd\n", __func__, reads, writes);
dbg_time("%s send fail filesend=%zd, filesize=%zd\n", __func__, filesend, filesize);
break;
}
filesend += reads;
// dbg_time("filesend=%zd, filesize=%zd\n", filesend, filesize);
}
else
{
break;
}
}
}
printf("\n");
show_progress();
dbg_time("send finished\n");
fclose(fp);
free(unix_filename);
free(pbuf);
if (is_upgrade_fimeware_zip_7z)
{
unlink(full_path);
}
if (filesend >= filesize) return 0;
error_return();
}
static int fh_process_erase(struct fh_data *fh_data, const struct fh_cmd *fh_cmd)
{
struct fh_cmd fh_rx_cmd;
unsigned timeout = 15000; // 8+8 MCP need more time
fh_send_cmd(fh_data, fh_cmd);
if (fh_wait_response_cmd(fh_data, &fh_rx_cmd, timeout) != 0) // SDX55 need 4 seconds
error_return();
if (strcmp(fh_rx_cmd.response.value, "ACK")) error_return();
return 0;
}
static int fh_process_patch(struct fh_data *fh_data, const struct fh_cmd *fh_cmd)
{
struct fh_cmd fh_rx_cmd;
unsigned timeout = 15000; // 8+8 MCP need more time
fh_send_cmd(fh_data, fh_cmd);
if (fh_wait_response_cmd(fh_data, &fh_rx_cmd, timeout) != 0) // SDX55 need 4 seconds
{
dbg_time("fh_process_patch : fh_wait_response_cmd fail\n");
error_return();
}
if (strcmp(fh_rx_cmd.response.value, "ACK"))
{
dbg_time("fh_process_patch : response should be ACK\n");
error_return();
}
return 0;
}
static int fh_process_sparse_program(struct fh_data *fh_data, const struct fh_cmd *fh_cmd)
{
char full_path[512];
char read_header_buf[64] = {0};
char read_chunk_header_buf[64] = {0};
char *ptmp;
FILE *fp;
size_t filesize /*, filesend*/;
char *unix_filename = strdup(fh_cmd->program.filename);
if (unix_filename == NULL) error_return();
void *pbuf = malloc(fh_data->MaxPayloadSizeToTargetInBytes);
if (pbuf == NULL)
{
if (unix_filename)
{
free(unix_filename);
unix_filename = NULL;
}
error_return();
}
memset(&SparseImgData, 0, sizeof(SparseImgParam));
while ((ptmp = strchr(unix_filename, '\\')))
{
*ptmp = '/';
}
snprintf(full_path, sizeof(full_path), "%.255s/%.240s", fh_data->firehose_dir, unix_filename);
fp = fopen(full_path, "rb");
if (!fp)
{
dbg_time("fail to fopen %s, errno: %d (%s)\n", full_path, errno, strerror(errno));
if (unix_filename)
{
free(unix_filename);
unix_filename = NULL;
}
if (pbuf)
{
free(pbuf);
pbuf = NULL;
}
error_return();
}
fseek(fp, 0, SEEK_END);
filesize = ftell(fp);
fseek(fp, 0, SEEK_SET);
dbg_time("send %s, filesize=%zd\n", unix_filename, filesize);
if (!strncasecmp(unix_filename, "gpt_empty0.bin", 14))
{
filesize -= 512;
}
uint32_t total_chunk_sz = 0;
uint32_t total_chunk_count = 0;
uint32_t total_sz = 0;
uint16_t chunk_type_Last_time = 0;
if (fh_cmd->program.sparse != NULL && !strncasecmp(fh_cmd->program.sparse, "true", 4))
{
sparse_header_t *sparse_header;
size_t read_header = fread(read_header_buf, 1, sizeof(sparse_header_t), fp);
if (read_header <= 0)
{
dbg_time("%s fread failed\n", __func__);
}
sparse_header = (sparse_header_t *)read_header_buf;
#if 0
printf("read_header:%ld\n", read_header);
printf("sparse_header->magic = %0x\n", sparse_header->magic);
printf("sparse_header->major_version = %0x\n", sparse_header->major_version);
printf("sparse_header->minor_version = %0x\n", sparse_header->minor_version);
printf("sparse_header->file_hdr_sz = %d\n", sparse_header->file_hdr_sz);
printf("sparse_header->chunk_hdr_sz = %d\n", sparse_header->chunk_hdr_sz);
printf("sparse_header->blk_sz = %d\n", sparse_header->blk_sz);
printf("sparse_header->total_blks = %d\n", sparse_header->total_blks);
printf("sparse_header->total_chunks = %d\n", sparse_header->total_chunks);
printf("sparse_header->image_checksum = %d\n", sparse_header->image_checksum);
#endif
SparseImgData.file_first_sector_offset = sparse_header->file_hdr_sz;
uint32_t m;
for (m = 0; m < (sparse_header->total_chunks); m++)
{
chunk_header_t *chunk_header;
read_header = fread(read_chunk_header_buf, 1, sizeof(chunk_header_t), fp);
chunk_header = (chunk_header_t *)read_chunk_header_buf;
#if 0
printf("chunk_header->chunk_type = %0x\n", chunk_header->chunk_type);
printf("chunk_header->reserved1 = %0x\n", chunk_header->reserved1);
printf("chunk_header->chunk_sz = %d\n", chunk_header->chunk_sz);
printf("chunk_header->total_sz = %d\n", chunk_header->total_sz);
#endif
if (chunk_header->chunk_type == 0xCAC1 || chunk_header->chunk_type == 0xCAC2)
{
if ((chunk_type_Last_time != 0xCAC1 && chunk_type_Last_time != 0xCAC2) && chunk_type_Last_time != 0)
{
SparseImgData.chunk_polymerization_cac3[SparseImgData.total_cac3_count].total_chunk_sz = total_chunk_sz;
SparseImgData.chunk_polymerization_cac3[SparseImgData.total_cac3_count].total_chunk_count = total_chunk_count;
SparseImgData.chunk_polymerization_cac3[SparseImgData.total_cac3_count].total_sz = total_sz;
SparseImgData.total_cac3_count += 1;
// printf("%s cac3 total_sz:%d total_chunk_count:d total_chunk_count:%d\n",
// __func__, total_sz, total_chunk_count, total_chunk_count);
total_chunk_sz = 0;
total_chunk_sz += chunk_header->chunk_sz;
total_chunk_count = 0; // count from 1
total_chunk_count += 1;
total_sz = 0;
total_sz += (chunk_header->total_sz - 0xC); // total data, out of size of chunk_header
fseek(fp, chunk_header->total_sz - 0xC, SEEK_CUR);
}
else if (chunk_type_Last_time == 0) // count from 1
{
total_chunk_sz = 0;
total_chunk_sz += chunk_header->chunk_sz;
total_chunk_count = 0;
total_chunk_count += 1;
total_sz = 0;
total_sz += (chunk_header->total_sz - 0xC);
fseek(fp, chunk_header->total_sz - 0xC, SEEK_CUR);
}
else
{
fseek(fp, chunk_header->total_sz - 0xC, SEEK_CUR);
total_sz += (chunk_header->total_sz - 0xC);
total_chunk_count += 1;
total_chunk_sz += chunk_header->chunk_sz;
}
}
if (chunk_header->chunk_type == 0xCAC3)
{
if (chunk_type_Last_time != 0xCAC3 && chunk_type_Last_time != 0)
{
SparseImgData.chunk_polymerization_data[SparseImgData.total_count].total_sz = total_sz;
SparseImgData.chunk_polymerization_data[SparseImgData.total_count].total_chunk_count = total_chunk_count;
SparseImgData.chunk_polymerization_data[SparseImgData.total_count].total_chunk_sz = total_chunk_sz;
SparseImgData.total_count += 1;
// printf("%s cac1+2 total_sz:%d total_chunk_count:%d total_chunk_count:%d\n",
// __func__, total_sz, total_chunk_count, total_chunk_count);
total_chunk_sz = 0;
total_chunk_sz += chunk_header->chunk_sz;
total_chunk_count = 0; // count from 1
total_chunk_count += 1;
total_sz = 0;
total_sz += (chunk_header->total_sz - 0xC);
fseek(fp, chunk_header->total_sz - 0xC, SEEK_CUR);
}
else
{
fseek(fp, chunk_header->total_sz - 0xC, SEEK_CUR);
total_sz += (chunk_header->total_sz - 0xC);
total_chunk_count += 1;
total_chunk_sz += chunk_header->chunk_sz;
// fseek(fp, chunk_header->total_sz - 0xC, SEEK_CUR);
// total_sz = 0;
// total_chunk_count = 0;
}
}
if (m == (sparse_header->total_chunks - 1) && (chunk_header->chunk_type == 0xCAC1 || chunk_header->chunk_type == 0xCAC2))
{
SparseImgData.chunk_polymerization_data[SparseImgData.total_count].total_sz = total_sz;
SparseImgData.chunk_polymerization_data[SparseImgData.total_count].total_chunk_count = total_chunk_count;
SparseImgData.chunk_polymerization_data[SparseImgData.total_count].total_chunk_sz = total_chunk_sz;
SparseImgData.total_count += 1;
// printf("%s cac1+2 total_sz:%d total_chunk_count:%d total_chunk_count:%d\n",
// __func__, total_sz, total_chunk_count, total_chunk_count);
}
chunk_type_Last_time = chunk_header->chunk_type;
}
}
fclose(fp);
free(unix_filename);
free(pbuf);
return 0;
}
static int fh_process_program(struct fh_data *fh_data, struct fh_cmd *fh_cmd)
{
struct fh_cmd fh_rx_cmd;
int i;
if (fh_cmd->program.sparse != NULL && !strncasecmp(fh_cmd->program.sparse, "true", 4))
{
fh_process_sparse_program(fh_data, fh_cmd);
for (i = 0; i < SparseImgData.total_count; i++)
{
if (i == 0)
{
fh_cmd->program.file_sector_offset = SparseImgData.file_first_sector_offset;
// printf("%s --1-- fh_cmd->program.file_sector_offset = %d\n", __func__,
// fh_cmd->program.file_sector_offset);
}
else
{
fh_cmd->program.file_sector_offset += SparseImgData.chunk_polymerization_data[i - 1].total_sz +
SparseImgData.chunk_polymerization_data[i - 1].total_chunk_count * sizeof(chunk_header_t) +
SparseImgData.chunk_polymerization_cac3[i - 1].total_chunk_count * sizeof(chunk_header_t);
// printf("%s --2-- fh_cmd->program.file_sector_offset = %d\n", __func__,
// fh_cmd->program.file_sector_offset);
}
if (i == 0)
{
; // printf("%s --1-- fh_cmd->program.start_sector = %d\n", __func__,
// fh_cmd->program.start_sector);
}
else
{
fh_cmd->program.start_sector += fh_cmd->program.num_partition_sectors + SparseImgData.chunk_polymerization_cac3[i - 1].total_chunk_sz * 8; //<2F><>Ҫ<EFBFBD><D2AA>ʼ+CAC1,CAC2,CAC3
// printf("%s --2-- fh_cmd->program.start_sector = %d\n", __func__,
// fh_cmd->program.start_sector);
}
fh_cmd->program.UNSPARSE_FILE_SIZE = SparseImgData.chunk_polymerization_data[i].total_sz;
fh_cmd->program.num_partition_sectors = fh_cmd->program.UNSPARSE_FILE_SIZE / fh_cmd->program.SECTOR_SIZE_IN_BYTES;
if (fh_cmd->program.UNSPARSE_FILE_SIZE % fh_cmd->program.SECTOR_SIZE_IN_BYTES) fh_cmd->program.num_partition_sectors += 1;
fh_send_cmd(fh_data, fh_cmd);
if (fh_wait_response_cmd(fh_data, &fh_rx_cmd, 3000) != 0)
{
dbg_time("fh_wait_response_cmd fail\n");
error_return();
}
if (strcmp(fh_rx_cmd.response.value, "ACK"))
{
dbg_time("response should be ACK\n");
error_return();
}
if (fh_rx_cmd.response.rawmode != 1)
{
dbg_time("response should be rawmode true\n");
error_return();
}
if (fh_send_rawmode_image(fh_data, fh_cmd, 15000))
{
dbg_time("fh_send_rawmode_image fail\n");
error_return();
}
if (fh_wait_response_cmd(fh_data, &fh_rx_cmd, 6000) != 0)
{
dbg_time("fh_wait_response_cmd fail\n");
error_return();
}
if (strcmp(fh_rx_cmd.response.value, "ACK"))
{
dbg_time("response should be ACK\n");
error_return();
}
if (fh_rx_cmd.response.rawmode != 0)
{
dbg_time("response should be rawmode false\n");
error_return();
}
}
memset(&SparseImgData, 0, sizeof(SparseImgParam));
}
else
{
fh_send_cmd(fh_data, fh_cmd);
if (fh_wait_response_cmd(fh_data, &fh_rx_cmd, 3000) != 0)
{
dbg_time("fh_wait_response_cmd fail\n");
error_return();
}
if (strcmp(fh_rx_cmd.response.value, "ACK"))
{
dbg_time("response should be ACK\n");
error_return();
}
if (fh_rx_cmd.response.rawmode != 1)
{
dbg_time("response should be rawmode true\n");
error_return();
}
if (fh_send_rawmode_image(fh_data, fh_cmd, 15000))
{
dbg_time("fh_send_rawmode_image fail\n");
error_return();
}
if (fh_wait_response_cmd(fh_data, &fh_rx_cmd, 6000) != 0)
{
dbg_time("fh_wait_response_cmd fail\n");
error_return();
}
if (strcmp(fh_rx_cmd.response.value, "ACK"))
{
dbg_time("response should be ACK\n");
error_return();
}
if (fh_rx_cmd.response.rawmode != 0)
{
dbg_time("response should be rawmode false\n");
error_return();
}
}
free(fh_cmd->program.filename);
return 0;
}
int firehose_main(const char *firehose_dir, void *usb_handle, unsigned qusb_zlp_mode)
{
unsigned x;
char rawprogram_full_path[512];
char *xmlfile_list[32];
char xmlfile_tmp[32];
unsigned xmlfile_cnt = 0;
struct fh_cmd fh_rx_cmd;
struct fh_data *fh_data;
long long filesizes = 0;
long filesize = 0;
unsigned max_num_partition_sectors = 0;
static pthread_t recv_cmd_tid;
int first_earse_and_last_programm_SBL = 0;
int rawprogram_unsparse_exist = 0;
fh_data = (struct fh_data *)malloc(sizeof(struct fh_data));
if (!fh_data) error_return();
memset(fh_data, 0x00, sizeof(struct fh_data));
fh_data->firehose_dir = firehose_dir;
fh_data->usb_handle = usb_handle;
fh_data->xml_tx_size = sizeof(fh_data->xml_tx_buf);
fh_data->xml_rx_size = sizeof(fh_data->xml_rx_buf);
fh_data->ZlpAwareHost = qusb_zlp_mode;
if (is_upgrade_fimeware_zip_7z)
{
int i;
for (i = 0; i < 32; i++)
{
xmlfile_list[i] = (char *)malloc(256);
if (xmlfile_list[i] == NULL)
{
dbg_time("%s xmlfile_list malloc failed\n", __func__);
error_return();
}
}
char rawprogram_patch_filename[128] = {0};
char rawprogram_patch_firehose_dir[256] = {0};
if (q_module_packet_sign)
{
for (x = 0; x < 10; x++)
{
snprintf(xmlfile_tmp, sizeof(xmlfile_tmp), "rawprogram%u_secboot",
x); // use rawprogram%u Adaptation rawprogram%u_xxx for AG215S-GLR
for (i = 0; i < file_name_b.file_name_count; i++)
{
if ((strstr(file_name_b.file_backup_c[i].zip_file_name_backup, xmlfile_tmp) && strstr(file_name_b.file_backup_c[i].zip_file_name_backup, ".xml")))
{
dbg_time("file_name_b.file_backup_c[i].zip_file_name_backup:%s\n", file_name_b.file_backup_c[i].zip_file_name_backup);
dbg_time("file_name_b.file_backup_c[i].zip_file_dir_backup:%s\n", file_name_b.file_backup_c[i].zip_file_dir_backup);
if (strstr(file_name_b.file_backup_c[i].zip_file_dir_backup, "update/firehose"))
{
memmove(rawprogram_patch_filename, file_name_b.file_backup_c[i].zip_file_name_backup, strlen(file_name_b.file_backup_c[i].zip_file_name_backup));
memmove(rawprogram_patch_firehose_dir, file_name_b.file_backup_c[i].zip_file_dir_backup, strlen(file_name_b.file_backup_c[i].zip_file_dir_backup));
break;
}
}
}
if (rawprogram_patch_filename[0] != '\0') // find rawprogram file
{
memset(zip_cmd_buf, 0, sizeof(zip_cmd_buf));
if (is_upgrade_fimeware_only_zip)
{
snprintf(zip_cmd_buf, sizeof(zip_cmd_buf), "unzip -o -q %.240s '*%.200s' -d /tmp/ > %s", firehose_dir, rawprogram_patch_firehose_dir, ZIP_PROCESS_INFO);
}
else
{
snprintf(zip_cmd_buf, sizeof(zip_cmd_buf), "7z x %.240s -o/tmp/ %.200s > %s", firehose_dir, rawprogram_patch_firehose_dir, ZIP_PROCESS_INFO);
}
dbg_time("%s zip_cmd_buf:%s\n", __func__, zip_cmd_buf);
if (-1 == system(zip_cmd_buf))
{
dbg_time("%s system return error\n", __func__);
for (i = 0; i < 32; i++)
{
if (xmlfile_list[i])
{
free(xmlfile_list[i]);
xmlfile_list[i] = NULL;
}
}
error_return();
}
usleep(1000);
memmove(xmlfile_list[xmlfile_cnt], rawprogram_patch_firehose_dir, 240);
dbg_time("xmlfile_list[xmlfile_cnt] = %s", xmlfile_list[xmlfile_cnt]);
xmlfile_cnt++;
}
}
}
else
{
for (x = 0; x < 10; x++)
{
snprintf(xmlfile_tmp, sizeof(xmlfile_tmp), "rawprogram_unsparse%u",
x); // smart SA885GAPNA
memset(rawprogram_patch_filename, 0, sizeof(rawprogram_patch_filename));
memset(rawprogram_patch_firehose_dir, 0, sizeof(rawprogram_patch_firehose_dir));
for (i = 0; i < file_name_b.file_name_count; i++)
{
if ((strstr(file_name_b.file_backup_c[i].zip_file_name_backup, xmlfile_tmp) && strstr(file_name_b.file_backup_c[i].zip_file_name_backup, ".xml")))
{
dbg_time("file_name_b.file_backup_c[i].zip_file_name_backup:%s\n", file_name_b.file_backup_c[i].zip_file_name_backup);
dbg_time("file_name_b.file_backup_c[i].zip_file_dir_backup:%s\n", file_name_b.file_backup_c[i].zip_file_dir_backup);
if (strstr(file_name_b.file_backup_c[i].zip_file_dir_backup, "update/firehose"))
{
memmove(rawprogram_patch_filename, file_name_b.file_backup_c[i].zip_file_name_backup, strlen(file_name_b.file_backup_c[i].zip_file_name_backup));
memmove(rawprogram_patch_firehose_dir, file_name_b.file_backup_c[i].zip_file_dir_backup, strlen(file_name_b.file_backup_c[i].zip_file_dir_backup));
break;
}
}
}
if (rawprogram_patch_filename[0] != '\0') // find rawprogram file
{
memset(zip_cmd_buf, 0, sizeof(zip_cmd_buf));
if (is_upgrade_fimeware_only_zip)
{
snprintf(zip_cmd_buf, sizeof(zip_cmd_buf), "unzip -o -q %.240s '*%.200s' -d /tmp/ > %s", firehose_dir, rawprogram_patch_firehose_dir, ZIP_PROCESS_INFO);
}
else
{
snprintf(zip_cmd_buf, sizeof(zip_cmd_buf), "7z x %.240s -o/tmp/ %.200s > %s", firehose_dir, rawprogram_patch_firehose_dir, ZIP_PROCESS_INFO);
}
dbg_time("%s zip_cmd_buf:%s\n", __func__, zip_cmd_buf);
if (-1 == system(zip_cmd_buf))
{
dbg_time("%s system return error\n", __func__);
for (i = 0; i < 32; i++)
{
if (xmlfile_list[i])
{
free(xmlfile_list[i]);
xmlfile_list[i] = NULL;
}
}
error_return();
}
usleep(1000);
memmove(xmlfile_list[xmlfile_cnt], rawprogram_patch_firehose_dir, 240);
dbg_time("xmlfile_list[xmlfile_cnt] = %s", xmlfile_list[xmlfile_cnt]);
xmlfile_cnt++;
rawprogram_unsparse_exist++;
}
}
if (rawprogram_unsparse_exist == 0)
{
memset(rawprogram_patch_filename, 0, sizeof(rawprogram_patch_filename));
memset(rawprogram_patch_firehose_dir, 0, sizeof(rawprogram_patch_firehose_dir));
for (i = 0; i < file_name_b.file_name_count; i++)
{
if (strstr(file_name_b.file_backup_c[i].zip_file_name_backup, "rawprogram_") && strstr(file_name_b.file_backup_c[i].zip_file_name_backup, ".xml"))
{
dbg_time("file_name_b.file_backup_c[i].zip_file_name_backup:%s\n", file_name_b.file_backup_c[i].zip_file_name_backup);
dbg_time("file_name_b.file_backup_c[i].zip_file_dir_backup:%s\n", file_name_b.file_backup_c[i].zip_file_dir_backup);
if (strstr(file_name_b.file_backup_c[i].zip_file_dir_backup, "update/firehose"))
{
memmove(rawprogram_patch_filename, file_name_b.file_backup_c[i].zip_file_name_backup, strlen(file_name_b.file_backup_c[i].zip_file_name_backup));
memmove(rawprogram_patch_firehose_dir, file_name_b.file_backup_c[i].zip_file_dir_backup, strlen(file_name_b.file_backup_c[i].zip_file_dir_backup));
break;
}
}
}
if (rawprogram_patch_filename[0] != '\0') // find rawprogram file
{
memset(zip_cmd_buf, 0, sizeof(zip_cmd_buf));
if (is_upgrade_fimeware_only_zip)
{
snprintf(zip_cmd_buf, sizeof(zip_cmd_buf), "unzip -o -q %.240s '*%.200s' -d /tmp/ > %s", firehose_dir, rawprogram_patch_firehose_dir, ZIP_PROCESS_INFO);
}
else
{
snprintf(zip_cmd_buf, sizeof(zip_cmd_buf), "7z x %.240s -o/tmp/ %.200s > %s", firehose_dir, rawprogram_patch_firehose_dir, ZIP_PROCESS_INFO);
}
dbg_time("%s zip_cmd_buf:%s\n", __func__, zip_cmd_buf);
if (-1 == system(zip_cmd_buf))
{
dbg_time("%s system return error\n", __func__);
for (i = 0; i < 32; i++)
{
if (xmlfile_list[i])
{
free(xmlfile_list[i]);
xmlfile_list[i] = NULL;
}
}
error_return();
}
usleep(1000);
memmove(xmlfile_list[xmlfile_cnt], rawprogram_patch_firehose_dir, 240);
dbg_time("xmlfile_list[xmlfile_cnt] = %s", xmlfile_list[xmlfile_cnt]);
xmlfile_cnt++;
}
}
memset(rawprogram_patch_filename, 0, sizeof(rawprogram_patch_filename));
memset(rawprogram_patch_firehose_dir, 0, sizeof(rawprogram_patch_firehose_dir));
for (i = 0; i < file_name_b.file_name_count; i++)
{
if (strstr(file_name_b.file_backup_c[i].zip_file_name_backup, "firehose-rawprogram") && strstr(file_name_b.file_backup_c[i].zip_file_name_backup, ".xml"))
{
dbg_time("file_name_b.file_backup_c[i].zip_file_name_backup:%s\n", file_name_b.file_backup_c[i].zip_file_name_backup);
dbg_time("file_name_b.file_backup_c[i].zip_file_dir_backup:%s\n", file_name_b.file_backup_c[i].zip_file_dir_backup);
if (strstr(file_name_b.file_backup_c[i].zip_file_dir_backup, "update/firehose"))
{
memmove(rawprogram_patch_filename, file_name_b.file_backup_c[i].zip_file_name_backup, strlen(file_name_b.file_backup_c[i].zip_file_name_backup));
memmove(rawprogram_patch_firehose_dir, file_name_b.file_backup_c[i].zip_file_dir_backup, strlen(file_name_b.file_backup_c[i].zip_file_dir_backup));
break;
}
}
}
if (rawprogram_patch_filename[0] != '\0') // find rawprogram file
{
memset(zip_cmd_buf, 0, sizeof(zip_cmd_buf));
if (is_upgrade_fimeware_only_zip)
{
snprintf(zip_cmd_buf, sizeof(zip_cmd_buf), "unzip -o -q %.240s '*%.200s' -d /tmp/ > %s", firehose_dir, rawprogram_patch_firehose_dir, ZIP_PROCESS_INFO);
}
else
{
snprintf(zip_cmd_buf, sizeof(zip_cmd_buf), "7z x %.240s -o/tmp/ %.200s > %s", firehose_dir, rawprogram_patch_firehose_dir, ZIP_PROCESS_INFO);
}
dbg_time("%s zip_cmd_buf:%s\n", __func__, zip_cmd_buf);
if (-1 == system(zip_cmd_buf))
{
dbg_time("%s system return error\n", __func__);
for (i = 0; i < 32; i++)
{
if (xmlfile_list[i])
{
free(xmlfile_list[i]);
xmlfile_list[i] = NULL;
}
}
error_return();
}
usleep(1000);
memmove(xmlfile_list[xmlfile_cnt], rawprogram_patch_firehose_dir, 240);
dbg_time("xmlfile_list[xmlfile_cnt] = %s", xmlfile_list[xmlfile_cnt]);
xmlfile_cnt++;
}
for (x = 0; x < 10; x++)
{
snprintf(xmlfile_tmp, sizeof(xmlfile_tmp), "rawprogram%u",
x); // use rawprogram%u Adaptation rawprogram%u_xxx
memset(rawprogram_patch_filename, 0, sizeof(rawprogram_patch_filename));
memset(rawprogram_patch_firehose_dir, 0, sizeof(rawprogram_patch_firehose_dir));
for (i = 0; i < file_name_b.file_name_count; i++)
{
if ((strstr(file_name_b.file_backup_c[i].zip_file_name_backup, xmlfile_tmp) && strstr(file_name_b.file_backup_c[i].zip_file_name_backup, ".xml")))
{
dbg_time("file_name_b.file_backup_c[i].zip_file_name_backup:%s\n", file_name_b.file_backup_c[i].zip_file_name_backup);
dbg_time("file_name_b.file_backup_c[i].zip_file_dir_backup:%s\n", file_name_b.file_backup_c[i].zip_file_dir_backup);
if (strstr(file_name_b.file_backup_c[i].zip_file_dir_backup, "update/firehose"))
{
memmove(rawprogram_patch_filename, file_name_b.file_backup_c[i].zip_file_name_backup, strlen(file_name_b.file_backup_c[i].zip_file_name_backup));
memmove(rawprogram_patch_firehose_dir, file_name_b.file_backup_c[i].zip_file_dir_backup, strlen(file_name_b.file_backup_c[i].zip_file_dir_backup));
break;
}
}
}
if (rawprogram_patch_filename[0] != '\0') // find rawprogram file
{
memset(zip_cmd_buf, 0, sizeof(zip_cmd_buf));
if (is_upgrade_fimeware_only_zip)
{
snprintf(zip_cmd_buf, sizeof(zip_cmd_buf), "unzip -o -q %.240s '*%.200s' -d /tmp/ > %s", firehose_dir, rawprogram_patch_firehose_dir, ZIP_PROCESS_INFO);
}
else
{
snprintf(zip_cmd_buf, sizeof(zip_cmd_buf), "7z x %.240s -o/tmp/ %.200s > %s", firehose_dir, rawprogram_patch_firehose_dir, ZIP_PROCESS_INFO);
}
dbg_time("%s zip_cmd_buf:%s\n", __func__, zip_cmd_buf);
if (-1 == system(zip_cmd_buf))
{
dbg_time("%s system return error\n", __func__);
for (i = 0; i < 32; i++)
{
if (xmlfile_list[i])
{
free(xmlfile_list[i]);
xmlfile_list[i] = NULL;
}
}
error_return();
}
usleep(1000);
memmove(xmlfile_list[xmlfile_cnt], rawprogram_patch_firehose_dir, 240);
dbg_time("xmlfile_list[xmlfile_cnt] = %s", xmlfile_list[xmlfile_cnt]);
xmlfile_cnt++;
}
}
}
memset(rawprogram_patch_filename, 0, sizeof(rawprogram_patch_filename));
memset(rawprogram_patch_firehose_dir, 0, sizeof(rawprogram_patch_firehose_dir));
for (i = 0; i < file_name_b.file_name_count; i++)
{
if ((strstr(file_name_b.file_backup_c[i].zip_file_name_backup, "patch_") && strstr(file_name_b.file_backup_c[i].zip_file_name_backup, ".xml")) ||
(strstr(file_name_b.file_backup_c[i].zip_file_name_backup, "patch-") && strstr(file_name_b.file_backup_c[i].zip_file_name_backup, ".xml")))
{
dbg_time("file_name_b.file_backup_c[i].zip_file_name_backup:%s\n", file_name_b.file_backup_c[i].zip_file_name_backup);
dbg_time("file_name_b.file_backup_c[i].zip_file_dir_backup:%s\n", file_name_b.file_backup_c[i].zip_file_dir_backup);
if (strstr(file_name_b.file_backup_c[i].zip_file_dir_backup, "update/firehose"))
{
memmove(rawprogram_patch_filename, file_name_b.file_backup_c[i].zip_file_name_backup, strlen(file_name_b.file_backup_c[i].zip_file_name_backup));
memmove(rawprogram_patch_firehose_dir, file_name_b.file_backup_c[i].zip_file_dir_backup, strlen(file_name_b.file_backup_c[i].zip_file_dir_backup));
break;
}
}
}
if (rawprogram_patch_filename[0] != '\0') // find patch file
{
memset(zip_cmd_buf, 0, sizeof(zip_cmd_buf));
if (is_upgrade_fimeware_only_zip)
{
snprintf(zip_cmd_buf, sizeof(zip_cmd_buf), "unzip -o -q %.240s '*%.200s' -d /tmp/ > %s", firehose_dir, rawprogram_patch_firehose_dir, ZIP_PROCESS_INFO);
}
else
{
snprintf(zip_cmd_buf, sizeof(zip_cmd_buf), "7z x %.240s -o/tmp/ %.200s > %s", firehose_dir, rawprogram_patch_firehose_dir, ZIP_PROCESS_INFO);
}
dbg_time("%s zip_cmd_buf:%s\n", __func__, zip_cmd_buf);
if (-1 == system(zip_cmd_buf))
{
dbg_time("%s system return error\n", __func__);
for (i = 0; i < 32; i++)
{
if (xmlfile_list[i])
{
free(xmlfile_list[i]);
xmlfile_list[i] = NULL;
}
}
error_return();
}
usleep(1000);
memmove(xmlfile_list[xmlfile_cnt], rawprogram_patch_firehose_dir, 240);
dbg_time("xmlfile_list[xmlfile_cnt] = %s", xmlfile_list[xmlfile_cnt]);
xmlfile_cnt++;
}
for (x = 0; x < 10; x++)
{
snprintf(xmlfile_tmp, sizeof(xmlfile_tmp), "patch%u.xml", x);
memset(rawprogram_patch_filename, 0, sizeof(rawprogram_patch_filename));
memset(rawprogram_patch_firehose_dir, 0, sizeof(rawprogram_patch_firehose_dir));
for (i = 0; i < file_name_b.file_name_count; i++)
{
if ((strstr(file_name_b.file_backup_c[i].zip_file_name_backup, xmlfile_tmp) && strstr(file_name_b.file_backup_c[i].zip_file_name_backup, ".xml")))
{
dbg_time("file_name_b.file_backup_c[i].zip_file_name_backup:%s\n", file_name_b.file_backup_c[i].zip_file_name_backup);
dbg_time("file_name_b.file_backup_c[i].zip_file_dir_backup:%s\n", file_name_b.file_backup_c[i].zip_file_dir_backup);
if (strstr(file_name_b.file_backup_c[i].zip_file_dir_backup, "update/firehose"))
{
memmove(rawprogram_patch_filename, file_name_b.file_backup_c[i].zip_file_name_backup, strlen(file_name_b.file_backup_c[i].zip_file_name_backup));
memmove(rawprogram_patch_firehose_dir, file_name_b.file_backup_c[i].zip_file_dir_backup, strlen(file_name_b.file_backup_c[i].zip_file_dir_backup));
break;
}
}
}
if (rawprogram_patch_filename[0] != '\0') // find patch file
{
memset(zip_cmd_buf, 0, sizeof(zip_cmd_buf));
if (is_upgrade_fimeware_only_zip)
{
snprintf(zip_cmd_buf, sizeof(zip_cmd_buf), "unzip -o -q %.240s '*%.200s' -d /tmp/ > %s", firehose_dir, rawprogram_patch_firehose_dir, ZIP_PROCESS_INFO);
}
else
{
snprintf(zip_cmd_buf, sizeof(zip_cmd_buf), "7z x %.240s -o/tmp/ %.200s > %s", firehose_dir, rawprogram_patch_firehose_dir, ZIP_PROCESS_INFO);
}
dbg_time("%s zip_cmd_buf:%s\n", __func__, zip_cmd_buf);
if (-1 == system(zip_cmd_buf))
{
dbg_time("%s system return error\n", __func__);
for (i = 0; i < 32; i++)
{
if (xmlfile_list[i])
{
free(xmlfile_list[i]);
xmlfile_list[i] = NULL;
}
}
error_return();
}
usleep(1000);
memmove(xmlfile_list[xmlfile_cnt], rawprogram_patch_firehose_dir, 240);
dbg_time("xmlfile_list[xmlfile_cnt] = %s", xmlfile_list[xmlfile_cnt]);
xmlfile_cnt++;
}
}
for (x = 0; x < xmlfile_cnt; x++)
{
snprintf(rawprogram_full_path, sizeof(rawprogram_full_path), "/tmp/%.255s", xmlfile_list[x]);
free(xmlfile_list[xmlfile_cnt]);
xmlfile_list[xmlfile_cnt] = NULL;
fh_parse_xml_file(fh_data, rawprogram_full_path);
unlink(rawprogram_full_path);
}
for (i = 0; i < 32; i++)
{
if (xmlfile_list[i])
{
free(xmlfile_list[i]);
xmlfile_list[i] = NULL;
}
}
if (fh_data->fh_cmd_count == 0)
{
if (fh_data)
{
free(fh_data);
fh_data = NULL;
}
error_return();
}
}
else
{
if (q_module_packet_sign)
{
for (x = 0; x < 10; x++)
{
snprintf(xmlfile_tmp, sizeof(xmlfile_tmp), "rawprogram%u_secboot",
x); // use rawprogram%u Adaptation rawprogram%u_xxx for AG215S-GLR
if (!qfile_find_file(firehose_dir, xmlfile_tmp, ".xml", &xmlfile_list[xmlfile_cnt]))
{
continue;
}
xmlfile_cnt++;
}
if (!qfile_find_file(firehose_dir, "secboot_", ".xml", &xmlfile_list[xmlfile_cnt]))
{
dbg_time("secboot rawprogram namd file failed.\n");
// error_return();
}
else
{
xmlfile_cnt++;
}
}
else
{
for (x = 0; x < 10; x++)
{
snprintf(xmlfile_tmp, sizeof(xmlfile_tmp), "rawprogram_unsparse%u",
x); // smart SA885GAPNA
if (!qfile_find_file(firehose_dir, xmlfile_tmp, ".xml", &xmlfile_list[xmlfile_cnt]))
{
continue;
}
xmlfile_cnt++;
rawprogram_unsparse_exist++;
}
if (rawprogram_unsparse_exist == 0)
{
if (!qfile_find_file(firehose_dir, "rawprogram_", ".xml", &xmlfile_list[xmlfile_cnt]))
{
dbg_time("retrieve rawprogram namd file failed.\n");
// error_return();
}
else
{
xmlfile_cnt++;
}
}
if (!qfile_find_file(firehose_dir, "firehose-rawprogram", ".xml", &xmlfile_list[xmlfile_cnt]))
{
dbg_time("retrieve rawprogram namd file failed.\n");
// error_return();
}
else
xmlfile_cnt++;
for (x = 0; x < 10; x++)
{
snprintf(xmlfile_tmp, sizeof(xmlfile_tmp), "rawprogram%u",
x); // use rawprogram%u Adaptation rawprogram%u_xxx
if (!qfile_find_file(firehose_dir, xmlfile_tmp, ".xml", &xmlfile_list[xmlfile_cnt]))
{
continue;
}
xmlfile_cnt++;
}
}
if (!qfile_find_file(firehose_dir, "patch_", ".xml", &xmlfile_list[xmlfile_cnt]) && !qfile_find_file(firehose_dir, "patch-", ".xml", &xmlfile_list[xmlfile_cnt]))
{
dbg_time("retrieve patch namd file failed.\n");
// error_return();
}
else
xmlfile_cnt++;
for (x = 0; x < 10; x++)
{
snprintf(xmlfile_tmp, sizeof(xmlfile_tmp), "patch%u.xml", x);
if (!qfile_find_file(firehose_dir, xmlfile_tmp, ".xml", &xmlfile_list[xmlfile_cnt]))
{
continue;
}
xmlfile_cnt++;
}
for (x = 0; x < xmlfile_cnt; x++)
{
snprintf(rawprogram_full_path, sizeof(rawprogram_full_path), "%.255s/%.255s", firehose_dir, xmlfile_list[x]);
free(xmlfile_list[xmlfile_cnt]);
fh_parse_xml_file(fh_data, rawprogram_full_path);
}
if (fh_data->fh_cmd_count == 0)
{
if (fh_data)
{
free(fh_data);
fh_data = NULL;
}
error_return();
}
}
for (x = 0; x < fh_data->fh_cmd_count; x++)
{
struct fh_cmd *fh_cmd = &fh_data->fh_cmd_table[x];
if (strstr(fh_cmd->cmd.type, "program"))
{
fh_fixup_program_cmd(fh_data, fh_cmd, &filesize);
if (fh_cmd->program.num_partition_sectors == 0)
{
if (fh_data)
{
free(fh_data);
fh_data = NULL;
}
error_return();
}
// calc files size
filesizes += filesize;
}
else if (strstr(fh_cmd->cmd.type, "erase"))
{
if ((fh_cmd->erase.num_partition_sectors + fh_cmd->erase.start_sector) > max_num_partition_sectors)
max_num_partition_sectors = (fh_cmd->erase.num_partition_sectors + fh_cmd->erase.start_sector);
}
}
if (socketpair(AF_LOCAL, SOCK_STREAM, 0, fh_recv_cmd_sk))
{
if (fh_data)
{
free(fh_data);
fh_data = NULL;
}
error_return();
}
fcntl(fh_recv_cmd_sk[0], F_SETFL, O_NONBLOCK);
if (pthread_create(&recv_cmd_tid, NULL, fh_recv_cmd_thread, (void *)fh_data)) error_return();
set_transfer_allbytes(filesizes);
// must first read <log from mdm9x07, then send <configure, and 1 second is not enough
fh_recv_cmd(fh_data, &fh_rx_cmd, 3000, 1);
while (fh_recv_cmd(fh_data, &fh_rx_cmd, 1000, 1) == 0)
;
if (fh_send_cfg_cmd(fh_data, q_device_type)) error_return();
if (!strcmp(q_device_type, "nand")) first_earse_and_last_programm_SBL = 1;
if (first_earse_and_last_programm_SBL || q_erase_all_before_download)
{
for (x = 0; x < fh_data->fh_cmd_count; x++)
{
struct fh_cmd *fh_cmd = &fh_data->fh_cmd_table[x];
if (!strstr(fh_cmd->cmd.type, "erase")) continue;
if (fh_cmd->erase.start_sector != 0) // Pre erase start_sector == 0 partition
continue;
if (q_erase_all_before_download)
{
fh_xml_set_value(fh_cmd->xml_original_data, "num_partition_sectors", max_num_partition_sectors);
if (fh_cmd->erase.last_sector)
{
fh_xml_set_value(fh_cmd->xml_original_data, "last_sector", max_num_partition_sectors - 1);
}
}
// dbg_time("point one");
if (fh_process_erase(fh_data, fh_cmd)) error_return();
}
}
for (x = 0; x < fh_data->fh_cmd_count; x++)
{
const struct fh_cmd *fh_cmd = &fh_data->fh_cmd_table[x];
if (strstr(fh_cmd->cmd.type, "vendor"))
{
fh_send_cmd(fh_data, fh_cmd);
if (fh_wait_response_cmd(fh_data, &fh_rx_cmd, 6000) != 0) error_return();
if (strcmp(fh_rx_cmd.response.value, "ACK")) error_return();
}
}
if (!q_erase_all_before_download)
{
for (x = 0; x < fh_data->fh_cmd_count; x++)
{
struct fh_cmd *fh_cmd = &fh_data->fh_cmd_table[x];
if (!strstr(fh_cmd->cmd.type, "erase")) continue;
if (fh_cmd->erase.SECTOR_SIZE_IN_BYTES == 0) //<2F><>ֹBG95 <20><><EFBFBD><EFBFBD>¼ jira id: STMDM9205-5237 <20><><EFBFBD><EFBFBD>fh_cmd->erase.num_partition_sectors ==
// 0<><30> <20><>Ϊ<erase SECTOR_SIZE_IN_BYTES="512" label="erase whole disk"
// physical_partition_number="0" start_sector="0" /> <20><>Ҫд<D2AA><D0B4>ģ<EFBFBD><C4A3>
continue;
if (first_earse_and_last_programm_SBL)
{
if (fh_cmd->erase.start_sector == 0) // Skip erase start_sector == 0 partition
continue;
}
// dbg_time("point two");
if (fh_process_erase(fh_data, fh_cmd)) error_return();
}
}
for (x = 0; x < fh_data->fh_cmd_count; x++)
{
struct fh_cmd *fh_cmd = &fh_data->fh_cmd_table[x];
if (!strstr(fh_cmd->cmd.type, "program")) continue;
if (first_earse_and_last_programm_SBL && fh_cmd->program.start_sector == 0) continue;
if (fh_process_program(fh_data, fh_cmd)) error_return();
}
if (first_earse_and_last_programm_SBL)
{
for (x = 0; x < fh_data->fh_cmd_count; x++)
{
struct fh_cmd *fh_cmd = &fh_data->fh_cmd_table[x];
if (!strstr(fh_cmd->cmd.type, "program")) continue;
if (fh_cmd->program.start_sector != 0) continue;
if (fh_process_program(fh_data, fh_cmd)) error_return();
}
}
if (fh_data->fh_patch_count)
{
for (x = 0; x < fh_data->fh_cmd_count; x++)
{
const struct fh_cmd *fh_cmd = &fh_data->fh_cmd_table[x];
if (!strstr(fh_cmd->cmd.type, "patch")) continue;
if (fh_process_patch(fh_data, fh_cmd)) error_return();
}
}
if (strcmp(q_device_type, "nand"))
{
fh_send_setbootablestoragedrive_cmd(fh_data);
if (fh_wait_response_cmd(fh_data, &fh_rx_cmd, 3000) != 0) error_return();
}
fh_send_reset_cmd(fh_data);
if (fh_wait_response_cmd(fh_data, &fh_rx_cmd, 3000) != 0) error_return();
while (fh_recv_cmd(fh_data, &fh_rx_cmd, 1000, 1) == 0)
; // required by sdx20
free(fh_data);
// pthread_join(recv_cmd_tid, NULL);
close(fh_recv_cmd_sk[0]);
close(fh_recv_cmd_sk[1]);
return 0;
}