Files
TencentOS-tiny/components/connectivity/iotkit-embedded-3.0.1/3rdparty/src/ota/iotx_ota.c
dcxajichu 8c24d921b0 support aliyun sdk on TencentOS tiny
sample: examples\aliyun_iotkit_csdk_mqtt
project: board\TencentOS_tiny_EVB_MX_Plus\KEIL\aliyun_iotkit_csdk_mqtt
2019-10-31 16:36:28 +08:00

985 lines
29 KiB
C

/*
* Copyright (C) 2015-2018 Alibaba Group Holding Limited
*/
#include "iotx_ota_internal.h"
#if (OTA_SIGNAL_CHANNEL) == 1
#include "ota_mqtt.c"
#elif (OTA_SIGNAL_CHANNEL) == 2
#include "ota_coap.c"
#else
#error "NOT support yet!"
#endif
typedef struct {
const char *product_key; /* point to product key */
const char *device_name; /* point to device name */
uint32_t id; /* message id */
IOT_OTA_State_t state; /* OTA state */
IOT_OTA_Type_t type; /* OTA Type */
uint32_t size_last_fetched; /* size of last downloaded */
uint32_t size_fetched; /* size of already downloaded */
uint32_t size_file; /* size of file */
char *purl; /* point to URL */
char *version; /* point to string */
char md5sum[33]; /* MD5 string */
void *md5; /* MD5 handle */
void *sha256; /* Sha256 handle */
void *ch_signal; /* channel handle of signal exchanged with OTA server */
void *ch_fetch; /* channel handle of download */
/* cota */
char *configId;
uint32_t configSize;
char *sign;
char *signMethod;
char *cota_url;
char *getType;
int err; /* last error code */
ota_fetch_cb_fpt fetch_cb; /* fetch_callback */
void *user_data; /* fetch_callback's user_data */
cota_fetch_cb_fpt fetch_cota_cb;
void *cota_user_data;
} OTA_Struct_t, *OTA_Struct_pt;
/* check whether the progress state is valid or not */
/* return: true, valid progress state; false, invalid progress state. */
static int ota_check_progress(IOT_OTA_Progress_t progress)
{
return ((progress >= IOT_OTAP_BURN_FAILED)
&& (progress <= IOT_OTAP_FETCH_PERCENTAGE_MAX));
}
int iotx_ota_set_fetch_callback(void *pt, ota_fetch_cb_fpt fetch_cb, void *user_data)
{
OTA_Struct_pt ota_pt = (OTA_Struct_pt)pt;
if (NULL == ota_pt || NULL == fetch_cb) {
return -1;
}
ota_pt->fetch_cb = fetch_cb;
ota_pt->user_data = user_data;
return 0;
}
int iotx_ota_set_cota_fetch_callback(void *pt, cota_fetch_cb_fpt fetch_cb, void *user_data)
{
OTA_Struct_pt ota_pt = (OTA_Struct_pt)pt;
if (NULL == ota_pt || NULL == fetch_cb) {
return -1;
}
ota_pt->fetch_cota_cb = fetch_cb;
ota_pt->cota_user_data = user_data;
return 0;
}
static int ota_callback(void *pcontext, const char *msg, uint32_t msg_len, iotx_ota_topic_types_t type)
{
const char *pvalue;
uint32_t val_len;
OTA_Struct_pt h_ota = (OTA_Struct_pt) pcontext;
if (h_ota->state == IOT_OTAS_FETCHING) {
OTA_LOG_INFO("In downloading state");
return -1;
}
switch (type) {
case IOTX_OTA_TOPIC_TYPE_DEVICE_REQUEST:
case IOTX_OTA_TOPIC_TYPE_DEVICE_UPGRATE: {
pvalue = otalib_JsonValueOf(msg, msg_len, "message", &val_len);
if (NULL == pvalue) {
OTA_LOG_ERROR("invalid json doc of OTA ");
return -1;
}
/* check whether is positive message */
if (!((strlen("success") == val_len) && (0 == strncmp(pvalue, "success", val_len)))) {
OTA_LOG_ERROR("fail state of json doc of OTA");
return -1;
}
/* get value of 'data' key */
pvalue = otalib_JsonValueOf(msg, msg_len, "data", &val_len);
if (NULL == pvalue) {
OTA_LOG_ERROR("Not 'data' key in json doc of OTA");
return -1;
}
if (0 != otalib_GetParams(pvalue, val_len, &h_ota->purl, &h_ota->version, h_ota->md5sum, &h_ota->size_file)) {
OTA_LOG_ERROR("Get config parameter failed");
return -1;
}
if (NULL == (h_ota->ch_fetch = ofc_Init(h_ota->purl))) {
OTA_LOG_ERROR("Initialize fetch module failed");
return -1;
}
h_ota->type = IOT_OTAT_FOTA;
h_ota->state = IOT_OTAS_FETCHING;
if (h_ota->fetch_cb) {
h_ota->fetch_cb(h_ota->user_data, 0, h_ota->size_file, h_ota->purl, h_ota->version);
}
}
break;
case IOTX_OTA_TOPIC_TYPE_CONFIG_GET: {
pvalue = otalib_JsonValueOf(msg, msg_len, "code", &val_len);
if (NULL == pvalue) {
OTA_LOG_ERROR("invalid json doc of OTA ");
return -1;
}
/* check whether is positive message */
if (!((strlen("200") == val_len) && (0 == strncmp(pvalue, "200", val_len)))) {
OTA_LOG_ERROR("fail state of json doc of OTA");
return -1;
}
/* get value of 'data' key */
pvalue = otalib_JsonValueOf(msg, msg_len, "data", &val_len);
if (NULL == pvalue) {
OTA_LOG_ERROR("Not 'data' key in json doc of OTA");
return -1;
}
if (0 != otalib_GetConfigParams(pvalue, val_len, &h_ota->configId, &h_ota->configSize,
&h_ota->sign, &h_ota->signMethod, &h_ota->cota_url, &h_ota->getType)) {
OTA_LOG_ERROR("Get firmware parameter failed");
return -1;
}
h_ota->size_file = h_ota->configSize;
h_ota->size_fetched = 0;
if (NULL != h_ota->md5) {
otalib_MD5Deinit(h_ota->md5);
}
h_ota->md5 = otalib_MD5Init();
if (NULL != h_ota->sha256) {
otalib_Sha256Deinit(h_ota->sha256);
}
h_ota->sha256 = otalib_Sha256Init();
if (NULL == (h_ota->ch_fetch = ofc_Init(h_ota->cota_url))) {
OTA_LOG_ERROR("Initialize fetch module failed");
return -1;
}
h_ota->type = IOT_OTAT_COTA;
h_ota->state = IOT_OTAS_FETCHING;
if (h_ota->fetch_cota_cb) {
h_ota->fetch_cota_cb(h_ota->user_data, 0, h_ota->configId, h_ota->configSize, h_ota->sign, h_ota->signMethod,
h_ota->cota_url, h_ota->getType);
}
}
break;
case IOTX_OTA_TOPIC_TYPE_CONFIG_PUSH: {
/* get value of 'params' key */
pvalue = otalib_JsonValueOf(msg, msg_len, "params", &val_len);
if (NULL == pvalue) {
OTA_LOG_ERROR("Not 'data' key in json doc of OTA");
return -1;
}
if (0 != otalib_GetConfigParams(pvalue, val_len, &h_ota->configId, &h_ota->configSize,
&h_ota->sign, &h_ota->signMethod, &h_ota->cota_url, &h_ota->getType)) {
OTA_LOG_ERROR("Get firmware parameter failed");
return -1;
}
h_ota->size_file = h_ota->configSize;
h_ota->size_fetched = 0;
if (NULL != h_ota->md5) {
otalib_MD5Deinit(h_ota->md5);
}
h_ota->md5 = otalib_MD5Init();
if (NULL != h_ota->sha256) {
otalib_Sha256Deinit(h_ota->sha256);
}
h_ota->sha256 = otalib_Sha256Init();
if (NULL == (h_ota->ch_fetch = ofc_Init(h_ota->cota_url))) {
OTA_LOG_ERROR("Initialize fetch module failed");
return -1;
}
h_ota->type = IOT_OTAT_COTA;
h_ota->state = IOT_OTAS_FETCHING;
if (h_ota->fetch_cota_cb) {
h_ota->fetch_cota_cb(h_ota->user_data, 0, h_ota->configId, h_ota->configSize, h_ota->sign, h_ota->signMethod,
h_ota->cota_url, h_ota->getType);
}
}
break;
default:
return -1;
break;
}
return 0;
}
static int g_ota_is_initialized = 0;
/* Initialize OTA module */
void *IOT_OTA_Init(const char *product_key, const char *device_name, void *ch_signal)
{
OTA_Struct_pt h_ota = NULL;
if (1 == g_ota_is_initialized) {
OTA_LOG_ERROR("iot ota has been initialized");
return NULL;
}
if ((NULL == product_key) || (NULL == device_name)) {
OTA_LOG_ERROR("one or more parameters is invalid");
return NULL;
}
if (NULL == (h_ota = OTA_MALLOC(sizeof(OTA_Struct_t)))) {
OTA_LOG_ERROR("allocate failed");
return NULL;
}
memset(h_ota, 0, sizeof(OTA_Struct_t));
h_ota->type = IOT_OTAT_NONE;
h_ota->state = IOT_OTAS_UNINITED;
h_ota->ch_signal = osc_Init(product_key, device_name, ch_signal, ota_callback, h_ota);
if (NULL == h_ota->ch_signal) {
OTA_LOG_ERROR("initialize signal channel failed");
goto do_exit;
}
h_ota->md5 = otalib_MD5Init();
if (NULL == h_ota->md5) {
OTA_LOG_ERROR("initialize md5 failed");
goto do_exit;
}
h_ota->sha256 = otalib_Sha256Init();
if (NULL == h_ota->sha256) {
OTA_LOG_ERROR("initialize sha256 failed");
goto do_exit;
}
h_ota->product_key = product_key;
h_ota->device_name = device_name;
h_ota->state = IOT_OTAS_INITED;
g_ota_is_initialized = 1;
return h_ota;
do_exit:
if (NULL != h_ota->ch_signal) {
osc_Deinit(h_ota->ch_signal);
}
if (NULL != h_ota->md5) {
otalib_MD5Deinit(h_ota->md5);
}
if (NULL != h_ota->sha256) {
otalib_Sha256Deinit(h_ota->sha256);
}
if (NULL != h_ota) {
OTA_FREE(h_ota);
}
return NULL;
#undef AOM_INFO_MSG_LEN
}
/* deinitialize OTA module */
int IOT_OTA_Deinit(void *handle)
{
OTA_Struct_pt h_ota = (OTA_Struct_pt) handle;
if (NULL == h_ota) {
OTA_LOG_ERROR("handle is NULL");
return IOT_OTAE_INVALID_PARAM;
}
if (IOT_OTAS_UNINITED == h_ota->state) {
OTA_LOG_ERROR("handle is uninitialized");
h_ota->err = IOT_OTAE_INVALID_STATE;
return -1;
}
g_ota_is_initialized = 0;
if (NULL != h_ota->ch_signal) {
osc_Deinit(h_ota->ch_signal);
}
if (NULL != h_ota->ch_fetch) {
ofc_Deinit(h_ota->ch_fetch);
}
if (NULL != h_ota->md5) {
otalib_MD5Deinit(h_ota->md5);
}
if (NULL != h_ota->sha256) {
otalib_Sha256Deinit(h_ota->sha256);
}
if (NULL != h_ota->purl) {
OTA_FREE(h_ota->purl);
}
if (NULL != h_ota->version) {
OTA_FREE(h_ota->version);
}
if (NULL != h_ota->configId) {
OTA_FREE(h_ota->configId);
}
if (NULL != h_ota->sign) {
OTA_FREE(h_ota->sign);
}
if (NULL != h_ota->signMethod) {
OTA_FREE(h_ota->signMethod);
}
if (NULL != h_ota->cota_url) {
OTA_FREE(h_ota->cota_url);
}
if (NULL != h_ota->getType) {
OTA_FREE(h_ota->getType);
}
OTA_FREE(h_ota);
return 0;
}
#define OTA_VERSION_STR_LEN_MIN (1)
#define OTA_VERSION_STR_LEN_MAX (32)
int IOT_OTA_ReportVersion(void *handle, const char *version)
{
#define MSG_INFORM_LEN (128)
int ret, len;
char *msg_informed;
OTA_Struct_pt h_ota = (OTA_Struct_pt) handle;
if ((NULL == h_ota) || (NULL == version)) {
OTA_LOG_ERROR("one or more invalid parameter");
return IOT_OTAE_INVALID_PARAM;
}
len = strlen(version);
if ((len < OTA_VERSION_STR_LEN_MIN) || (len > OTA_VERSION_STR_LEN_MAX)) {
OTA_LOG_ERROR("version string is invalid: must be [1, 32] chars");
h_ota->err = IOT_OTAE_INVALID_PARAM;
return -1;
}
if (IOT_OTAS_UNINITED == h_ota->state) {
OTA_LOG_ERROR("handle is uninitialized");
h_ota->err = IOT_OTAE_INVALID_STATE;
return -1;
}
if (NULL == (msg_informed = OTA_MALLOC(MSG_INFORM_LEN))) {
OTA_LOG_ERROR("allocate for msg_informed failed");
h_ota->err = IOT_OTAE_NOMEM;
return -1;
}
ret = otalib_GenInfoMsg(msg_informed, MSG_INFORM_LEN, h_ota->id, version);
if (ret != 0) {
OTA_LOG_ERROR("generate inform message failed");
h_ota->err = ret;
ret = -1;
goto do_exit;
}
ret = osc_ReportVersion(h_ota->ch_signal, msg_informed);
if (0 != ret) {
OTA_LOG_ERROR("Report version failed");
h_ota->err = ret;
ret = -1;
goto do_exit;
}
ret = 0;
do_exit:
if (NULL != msg_informed) {
OTA_FREE(msg_informed);
}
return ret;
#undef MSG_INFORM_LEN
}
int iotx_req_image(void *handle, const char *version)
{
#define MSG_REQUEST_LEN (128)
int ret, len;
char *msg_informed;
OTA_Struct_pt h_ota = (OTA_Struct_pt) handle;
if ((NULL == h_ota) || (NULL == version)) {
OTA_LOG_ERROR("one or more invalid parameter");
return IOT_OTAE_INVALID_PARAM;
}
len = strlen(version);
if ((len < OTA_VERSION_STR_LEN_MIN) || (len > OTA_VERSION_STR_LEN_MAX)) {
OTA_LOG_ERROR("version string is invalid: must be [1, 32] chars");
h_ota->err = IOT_OTAE_INVALID_PARAM;
return -1;
}
if (IOT_OTAS_UNINITED == h_ota->state) {
OTA_LOG_ERROR("handle is uninitialized");
h_ota->err = IOT_OTAE_INVALID_STATE;
return -1;
}
if (IOT_OTAS_FETCHING == h_ota->state) {
OTA_LOG_ERROR("ota is busying");
h_ota->err = IOT_OTAE_INVALID_STATE;
return -1;
}
if (NULL == (msg_informed = OTA_MALLOC(MSG_REQUEST_LEN))) {
OTA_LOG_ERROR("allocate for msg_informed failed");
h_ota->err = IOT_OTAE_NOMEM;
return -1;
}
ret = otalib_GenInfoMsg(msg_informed, MSG_REQUEST_LEN, h_ota->id, version);
if (ret != 0) {
OTA_LOG_ERROR("generate request image message failed");
h_ota->err = ret;
ret = -1;
goto do_exit;
}
ret = osc_RequestImage(h_ota->ch_signal, msg_informed);
if (0 != ret) {
OTA_LOG_ERROR("Request image failed");
h_ota->err = ret;
ret = -1;
goto do_exit;
}
ret = 0;
do_exit:
if (NULL != msg_informed) {
OTA_FREE(msg_informed);
}
return ret;
#undef MSG_REQUEST_LEN
}
int IOT_OTA_ReportProgress(void *handle, IOT_OTA_Progress_t progress, const char *msg)
{
#define MSG_REPORT_LEN (256)
int ret = -1;
char *msg_reported;
OTA_Struct_pt h_ota = (OTA_Struct_pt) handle;
if (NULL == handle) {
OTA_LOG_ERROR("handle is NULL");
return IOT_OTAE_INVALID_PARAM;
}
if (IOT_OTAS_UNINITED == h_ota->state) {
OTA_LOG_ERROR("handle is uninitialized");
h_ota->err = IOT_OTAE_INVALID_STATE;
return -1;
}
if (!ota_check_progress(progress)) {
OTA_LOG_ERROR("progress is a invalid parameter");
h_ota->err = IOT_OTAE_INVALID_PARAM;
return -1;
}
if (NULL == (msg_reported = OTA_MALLOC(MSG_REPORT_LEN))) {
OTA_LOG_ERROR("allocate for msg_reported failed");
h_ota->err = IOT_OTAE_NOMEM;
return -1;
}
ret = otalib_GenReportMsg(msg_reported, MSG_REPORT_LEN, h_ota->id, progress, msg);
if (0 != ret) {
OTA_LOG_ERROR("generate reported message failed");
h_ota->err = ret;
goto do_exit;
}
ret = osc_ReportProgress(h_ota->ch_signal, msg_reported);
if (0 != ret) {
OTA_LOG_ERROR("Report progress failed");
h_ota->err = ret;
goto do_exit;
}
ret = 0;
do_exit:
if (NULL != msg_reported) {
OTA_FREE(msg_reported);
}
return ret;
#undef MSG_REPORT_LEN
}
int iotx_ota_get_config(void *handle, const char *configScope, const char *getType, const char *attributeKeys)
{
#define MSG_REPORT_LEN (256)
int ret = -1;
char *msg_get;
char topic[OTA_MQTT_TOPIC_LEN] = {0};
OTA_Struct_pt h_ota = (OTA_Struct_pt) handle;
iotx_mqtt_topic_info_t topic_info;
memset(&topic_info, 0, sizeof(iotx_mqtt_topic_info_t));
if (NULL == handle) {
OTA_LOG_ERROR("handle is NULL");
return IOT_OTAE_INVALID_PARAM;
}
if (IOT_OTAS_UNINITED == h_ota->state) {
OTA_LOG_ERROR("handle is uninitialized");
h_ota->err = IOT_OTAE_INVALID_STATE;
return -1;
}
if (IOT_OTAS_FETCHING == h_ota->state) {
OTA_LOG_ERROR("ota is busying");
h_ota->err = IOT_OTAE_INVALID_STATE;
return -1;
}
if (NULL == (msg_get = OTA_MALLOC(MSG_REPORT_LEN))) {
OTA_LOG_ERROR("allocate for msg_reported failed");
h_ota->err = IOT_OTAE_NOMEM;
return -1;
}
if (0 > HAL_Snprintf(topic,
OTA_MQTT_TOPIC_LEN,
"/sys/%s/%s/thing/config/get",
h_ota->product_key,
h_ota->device_name)) {
goto do_exit;
};
if (0 > HAL_Snprintf(msg_get,
MSG_REPORT_LEN,
"{\"id\" : %d,\"version\":\"1.0\",\"params\":{\"configScope\":\"%s\",\"getType\":\"%s\",\"attributeKeys\":\"%s\"},\"method\":\"thing.config.get\"}",
h_ota->id,
configScope,
getType,
attributeKeys)) {
goto do_exit;
};
OTA_LOG_INFO(msg_get);
topic_info.qos = IOTX_MQTT_QOS0;
topic_info.payload = (void *)msg_get;
topic_info.payload_len = strlen(msg_get);
ret = osc_RequestConfig(h_ota->ch_signal, topic, &topic_info);
if (ret < 0) {
OTA_LOG_ERROR("publish failed");
return IOT_OTAE_OSC_FAILED;
}
ret = 0;
do_exit:
if (NULL != msg_get) {
OTA_FREE(msg_get);
}
return ret;
#undef MSG_REPORT_LEN
}
/* check whether is downloading */
int IOT_OTA_IsFetching(void *handle)
{
OTA_Struct_pt h_ota = (OTA_Struct_pt)handle;
if (NULL == handle) {
OTA_LOG_ERROR("handle is NULL");
return 0;
}
if (IOT_OTAS_UNINITED == h_ota->state) {
OTA_LOG_ERROR("handle is uninitialized");
h_ota->err = IOT_OTAE_INVALID_STATE;
return 0;
}
return (IOT_OTAS_FETCHING == h_ota->state);
}
/* check whether fetch over */
int IOT_OTA_IsFetchFinish(void *handle)
{
OTA_Struct_pt h_ota = (OTA_Struct_pt) handle;
if (NULL == handle) {
OTA_LOG_ERROR("handle is NULL");
return 0;
}
if (IOT_OTAS_UNINITED == h_ota->state) {
OTA_LOG_ERROR("handle is uninitialized");
h_ota->err = IOT_OTAE_INVALID_STATE;
return 0;
}
return (IOT_OTAS_FETCHED == h_ota->state);
}
int IOT_OTA_FetchYield(void *handle, char *buf, uint32_t buf_len, uint32_t timeout_s)
{
int ret;
OTA_Struct_pt h_ota = (OTA_Struct_pt) handle;
if ((NULL == handle) || (NULL == buf) || (0 == buf_len)) {
OTA_LOG_ERROR("invalid parameter");
return IOT_OTAE_INVALID_PARAM;
}
if (IOT_OTAS_FETCHING != h_ota->state) {
h_ota->err = IOT_OTAE_INVALID_STATE;
return IOT_OTAE_INVALID_STATE;
}
ret = ofc_Fetch(h_ota->ch_fetch, buf, buf_len, timeout_s);
if (ret < 0) {
OTA_LOG_ERROR("Fetch firmware failed");
h_ota->state = IOT_OTAS_FETCHED;
h_ota->type = IOT_OTAT_NONE;
h_ota->err = IOT_OTAE_FETCH_FAILED;
if (h_ota->fetch_cb && h_ota->purl) {
h_ota->fetch_cb(h_ota->user_data, 1, h_ota->size_file, h_ota->purl, h_ota->version);
/* remove */
h_ota->purl = NULL;
} else if (h_ota->fetch_cota_cb && h_ota->cota_url) {
h_ota->fetch_cota_cb(h_ota->user_data, 1, h_ota->configId, h_ota->configSize, h_ota->sign, h_ota->signMethod,
h_ota->cota_url, h_ota->getType);
/* remove */
h_ota->cota_url = NULL;
}
h_ota->size_fetched = 0;
return -1;
} else if (0 == h_ota->size_fetched) {
/* force report status in the first */
IOT_OTA_ReportProgress(h_ota, IOT_OTAP_FETCH_PERCENTAGE_MIN, "Enter in downloading state");
}
otalib_MD5Update(h_ota->md5, buf, ret);
otalib_Sha256Update(h_ota->sha256, buf, ret);
h_ota->size_last_fetched = ret;
h_ota->size_fetched += ret;
if (h_ota->size_fetched >= h_ota->size_file) {
h_ota->type = IOT_OTAT_NONE;
h_ota->state = IOT_OTAS_FETCHED;
if (h_ota->fetch_cb && h_ota->purl) {
h_ota->fetch_cb(h_ota->user_data, 1, h_ota->size_file, h_ota->purl, h_ota->version);
/* remove */
h_ota->purl = NULL;
} else if (h_ota->fetch_cota_cb && h_ota->cota_url) {
h_ota->fetch_cota_cb(h_ota->user_data, 1, h_ota->configId, h_ota->configSize, h_ota->sign, h_ota->signMethod,
h_ota->cota_url, h_ota->getType);
/* remove */
h_ota->cota_url = NULL;
}
}
return ret;
}
int IOT_OTA_Ioctl(void *handle, IOT_OTA_CmdType_t type, void *buf, int buf_len)
{
OTA_Struct_pt h_ota = (OTA_Struct_pt) handle;
if ((NULL == handle) || (NULL == buf) || (0 == buf_len)) {
OTA_LOG_ERROR("invalid parameter");
return IOT_OTAE_INVALID_PARAM;
}
if (h_ota->state < IOT_OTAS_FETCHING) {
h_ota->err = IOT_OTAE_INVALID_STATE;
return IOT_OTAE_INVALID_STATE;
}
switch (type) {
case IOT_OTAG_COTA_CONFIG_ID: {
char **value = (char **)buf;
if (value == NULL || *value != NULL || h_ota->configId == NULL) {
OTA_LOG_ERROR("Invalid parameter");
h_ota->err = IOT_OTAE_INVALID_PARAM;
return -1;
} else {
*value = OTA_API_MALLOC(strlen(h_ota->configId) + 1);
if (*value == NULL) {
h_ota->err = IOT_OTAE_INVALID_PARAM;
return -1;
}
memset(*value, 0, strlen(h_ota->configId) + 1);
memcpy(*value, h_ota->configId, strlen(h_ota->configId));
return 0;
}
}
break;
case IOT_OTAG_COTA_CONFIG_SIZE: {
if ((4 != buf_len) || (0 != ((unsigned long)buf & 0x3))) {
OTA_LOG_ERROR("Invalid parameter");
h_ota->err = IOT_OTAE_INVALID_PARAM;
return -1;
} else {
*((uint32_t *)buf) = h_ota->configSize;
return 0;
}
}
break;
case IOT_OTAG_COTA_SIGN: {
char **value = (char **)buf;
if (value == NULL || *value != NULL || h_ota->sign == NULL) {
OTA_LOG_ERROR("Invalid parameter");
h_ota->err = IOT_OTAE_INVALID_PARAM;
return -1;
} else {
*value = OTA_API_MALLOC(strlen(h_ota->sign) + 1);
if (*value == NULL) {
h_ota->err = IOT_OTAE_INVALID_PARAM;
return -1;
}
memset(*value, 0, strlen(h_ota->sign) + 1);
memcpy(*value, h_ota->sign, strlen(h_ota->sign));
return 0;
}
}
break;
case IOT_OTAG_COTA_SIGN_METHOD: {
char **value = (char **)buf;
if (value == NULL || *value != NULL || h_ota->signMethod == NULL) {
OTA_LOG_ERROR("Invalid parameter");
h_ota->err = IOT_OTAE_INVALID_PARAM;
return -1;
} else {
*value = OTA_API_MALLOC(strlen(h_ota->signMethod) + 1);
if (*value == NULL) {
h_ota->err = IOT_OTAE_INVALID_PARAM;
return -1;
}
memset(*value, 0, strlen(h_ota->signMethod) + 1);
memcpy(*value, h_ota->signMethod, strlen(h_ota->signMethod));
return 0;
}
}
break;
case IOT_OTAG_COTA_URL: {
char **value = (char **)buf;
if (value == NULL || *value != NULL || h_ota->cota_url == NULL) {
OTA_LOG_ERROR("Invalid parameter");
h_ota->err = IOT_OTAE_INVALID_PARAM;
return -1;
} else {
*value = OTA_API_MALLOC(strlen(h_ota->cota_url) + 1);
if (*value == NULL) {
h_ota->err = IOT_OTAE_INVALID_PARAM;
return -1;
}
memset(*value, 0, strlen(h_ota->cota_url) + 1);
memcpy(*value, h_ota->cota_url, strlen(h_ota->cota_url));
return 0;
}
}
break;
case IOT_OTAG_COTA_GETTYPE: {
char **value = (char **)buf;
if (value == NULL || *value != NULL || h_ota->getType == NULL) {
OTA_LOG_ERROR("Invalid parameter");
h_ota->err = IOT_OTAE_INVALID_PARAM;
return -1;
} else {
*value = OTA_API_MALLOC(strlen(h_ota->getType) + 1);
if (*value == NULL) {
h_ota->err = IOT_OTAE_INVALID_PARAM;
return -1;
}
memset(*value, 0, strlen(h_ota->getType) + 1);
memcpy(*value, h_ota->getType, strlen(h_ota->getType));
return 0;
}
}
break;
case IOT_OTAG_OTA_TYPE: {
if ((4 != buf_len) || (0 != ((unsigned long)buf & 0x3))) {
OTA_LOG_ERROR("Invalid parameter");
h_ota->err = IOT_OTAE_INVALID_PARAM;
return -1;
} else {
*((uint32_t *)buf) = h_ota->type;
return 0;
}
}
break;
case IOT_OTAG_FETCHED_SIZE:
if ((4 != buf_len) || (0 != ((unsigned long)buf & 0x3))) {
OTA_LOG_ERROR("Invalid parameter");
h_ota->err = IOT_OTAE_INVALID_PARAM;
return -1;
} else {
*((uint32_t *)buf) = h_ota->size_fetched;
return 0;
}
case IOT_OTAG_FILE_SIZE:
if ((4 != buf_len) || (0 != ((unsigned long)buf & 0x3))) {
OTA_LOG_ERROR("Invalid parameter");
h_ota->err = IOT_OTAE_INVALID_PARAM;
return -1;
} else {
*((uint32_t *)buf) = h_ota->size_file;
return 0;
};
case IOT_OTAG_VERSION: {
strncpy(buf, h_ota->version, buf_len);
((char *)buf)[buf_len - 1] = '\0';
}
break;
case IOT_OTAG_MD5SUM:
strncpy(buf, h_ota->md5sum, buf_len);
((char *)buf)[buf_len - 1] = '\0';
break;
case IOT_OTAG_CHECK_FIRMWARE:
if ((4 != buf_len) || (0 != ((unsigned long)buf & 0x3))) {
OTA_LOG_ERROR("Invalid parameter");
h_ota->err = IOT_OTAE_INVALID_PARAM;
return -1;
} else if (h_ota->state != IOT_OTAS_FETCHED) {
h_ota->err = IOT_OTAE_INVALID_STATE;
OTA_LOG_ERROR("Firmware can be checked in IOT_OTAS_FETCHED state only");
return -1;
} else {
char md5_str[33];
otalib_MD5Finalize(h_ota->md5, md5_str);
OTA_LOG_DEBUG("origin=%s, now=%s", h_ota->md5sum, md5_str);
if (0 == strcmp(h_ota->md5sum, md5_str)) {
*((uint32_t *)buf) = 1;
} else {
*((uint32_t *)buf) = 0;
IOT_OTA_ReportProgress(h_ota, IOT_OTAP_CHECK_FALIED, NULL);
OTA_LOG_ERROR("image checksum compare failed");
}
return 0;
}
case IOT_OTAG_CHECK_CONFIG:
if ((4 != buf_len) || (0 != ((unsigned long)buf & 0x3))) {
OTA_LOG_ERROR("Invalid parameter");
h_ota->err = IOT_OTAE_INVALID_PARAM;
return -1;
} else if (h_ota->state != IOT_OTAS_FETCHED) {
h_ota->err = IOT_OTAE_INVALID_STATE;
OTA_LOG_ERROR("Config can be checked in IOT_OTAS_FETCHED state only");
return -1;
} else {
if (0 == strncmp(h_ota->signMethod, "Md5", strlen(h_ota->signMethod))) {
char md5_str[33];
otalib_MD5Finalize(h_ota->md5, md5_str);
OTA_LOG_DEBUG("origin=%s, now=%s", h_ota->sign, md5_str);
if (0 == strcmp(h_ota->sign, md5_str)) {
*((uint32_t *)buf) = 1;
} else {
*((uint32_t *)buf) = 0;
}
}
if (0 == strncmp(h_ota->signMethod, "Sha256", strlen(h_ota->signMethod))) {
char sha256_str[65];
otalib_Sha256Finalize(h_ota->sha256, sha256_str);
OTA_LOG_DEBUG("origin=%s, now=%s", h_ota->sign, sha256_str);
if (0 == strcmp(h_ota->sign, sha256_str)) {
*((uint32_t *)buf) = 1;
} else {
*((uint32_t *)buf) = 0;
}
}
return 0;
}
case IOT_OTAG_RESET_FETCHED_SIZE: {
h_ota->size_fetched = 0;
return 0;
}
default:
OTA_LOG_ERROR("invalid cmd type");
h_ota->err = IOT_OTAE_INVALID_PARAM;
return -1;
}
return 0;
}
/* Get last error code */
int IOT_OTA_GetLastError(void *handle)
{
OTA_Struct_pt h_ota = (OTA_Struct_pt) handle;
if (NULL == handle) {
OTA_LOG_ERROR("handle is NULL");
return IOT_OTAE_INVALID_PARAM;
}
return h_ota->err;
}