# 固件升級示例-FirmwareUpgrade
功能描述:本示例演示如何用固件文件給設備升級。
> 本示例基于C Low Level API進行演示,示例編譯語言為C++,OrbbecSDK使用C語言API
在main函數(shù)接口通過命令參數(shù)獲取固件文件
// check_firmware_file_path()函數(shù)用于檢查文件是否存在,實際代碼中最好檢查后綴是否為bin或者img, 以及固件文件是否與目標設備相匹配
const char *check_firmware_file_path(int argc, char **argv) {
if(argc < 2) {
printf("Please input firmware path.\n");
return "";
}
const char *filePath = *(argv + 1);
FILE *file = fopen(filePath, "r");
if(!file) {
printf("Open Firmware file failed. filePath: %s\n", filePath);
return "";
}
fclose(file);
return filePath;
}
int main(int argc, char **argv) {
const char *firmware_file_path = check_firmware_file_path(argc, argv);
if(!firmware_file_path || 0 == strlen(firmware_file_path)) {
printf("command: \n$ ./frameware_upgrade[.exe] firmwareFile.bin\n");
return 0;
}
// 接下來的業(yè)務代碼
return 0;
}
創(chuàng)建ob_context并通過ob_context獲取設備,本示例假設運行firmware_upgrade[.exe]之前,上位機(Windows、Ubuntu、Arm平臺)已經(jīng)插入設備。device_changed_callback用于固件升級后監(jiān)聽device重啟后獲取被升級設備的業(yè)務處理
// 構(gòu)建ob_context對象
ob_error *error = NULL;
ob_context *ctx = ob_create_context(&error);
check_error(error);
// 設置設備變化監(jiān)聽器,device_changed_callback是管理device聲明周期的關(guān)鍵函數(shù),開發(fā)者必須關(guān)注該回調(diào)
ob_set_device_changed_callback(ctx, device_changed_callback, &callback_user_data_, &error);
check_error(error);
// 查詢當前已經(jīng)接入的設備
ob_device_list *dev_list = ob_query_device_list(ctx, &error);
check_error(error);
// 從ob_device_list中獲取當前接入設備的數(shù)量
int dev_count = ob_device_list_device_count(dev_list, &error);
check_error(error);
if(dev_count == 0) {
// 固件升級示例假設設備已經(jīng)接入到上位機(Windows、Ubuntu、Android平臺)
printf("Device not found!\n");
return -1;
}
// 獲取第一個設備,index=0
ob_device *dev = ob_device_list_get_device(dev_list, 0, &error);
check_error(error);
// 打印設備信息
dump_device_info(dev);
獲取設備當前的固件版本信息
// 打印設備名稱,SN,VID,PID以及固件版本
void dump_device_info(ob_device *device) {
ob_error *error = NULL;
// 獲取ob_device_info對象,通過ob_device_info可以獲取目標設備的基本信息
ob_device_info *dev_info = ob_device_get_device_info(device, &error);
check_error(error);
// 設備名稱
const char *name = ob_device_info_name(dev_info, &error);
check_error(error);
printf("Device name: %s\n", name);
// 設備VID,PID,UID
int pid = ob_device_info_pid(dev_info, &error);
check_error(error);
int vid = ob_device_info_vid(dev_info, &error);
check_error(error);
const char *uid = ob_device_info_uid(dev_info, &error);
check_error(error);
printf("Device pid: %d vid: %d uid: %s\n", pid, vid, uid);
// 設備當前的固件版本號
const char *fw_ver = ob_device_info_firmware_version(dev_info, &error);
check_error(error);
printf("Firmware version: %s\n", fw_ver);
// 設備SN
const char *sn = ob_device_info_serial_number(dev_info, &error);
check_error(error);
printf("Serial number: %s\n", sn);
// 釋放資源,否則會造成內(nèi)存泄漏
ob_delete_device_info(dev_info, &error);
check_error(error);
}
給目標設備升級固件
a. 實現(xiàn)C API的固件升級回調(diào)接口;
b. 調(diào)用固件升級接口進行升級;
// 實現(xiàn)C API的固件升級回調(diào)接口;
void device_upgrade_callback(ob_upgrade_state state, const char *message, uint8_t percent, void *user_data) {
if(state == STAT_START) {
printf("Upgrade Firmware start\n");
}
else if(state == STAT_FILE_TRANSFER) {
printf("Upgrade Firmware file transfer, percent: %u\n", (uint32_t)percent);
}
else if(state == STAT_IN_PROGRESS) {
printf("Upgrade Firmware in progress, percent: %u\n", (uint32_t)percent);
}
else if(state == STAT_DONE) {
// 固件升級成功
printf("Upgrade Firmware done, percent: %u\n", (uint32_t)percent);
is_upgrade_success_ = true;
}
else if(state == STAT_VERIFY_IMAGE) {
printf("Upgrade Firmware verify image\n");
}
else {
// 固件升級失敗
printf("Upgrade Firmware failed. state: %d, errMsg: %s, percent: %u \n", (int)state, message ? message : "", (uint32_t)percent);
}
}
// 對目標設備進行固件升級
bool upgrade_firmware(ob_device *device, const char *firmwarePath) {
const char *index = strstr(firmwarePath, ".img");
bool isImgFile = (bool)index;
index = strstr(firmwarePath, ".bin");
bool isBinFile = (bool)index;
if(!(isImgFile || isBinFile)) {
// 固件升級文件一般為bin或者img,實際業(yè)務中最好通過文件名稱、文件MD5等信息做防呆
printf("Upgrade Fimware failed. invalid firmware file: %s\n", firmwarePath);
return false;
}
// 調(diào)用固件升級接口進行升級;
is_upgrade_success_ = false;
ob_error *error = NULL;
ob_device_upgrade(device, firmwarePath, device_upgrade_callback, false, &callback_user_data_, &error);
check_error(error);
return is_upgrade_success_;
}
固件升級成功后,需要重啟設備。重啟設備有兩種方式,一種是拔插設備(或者操作系統(tǒng)重啟),另一種是調(diào)用OrbbecSDK的reboot接口。等設備上線后,通過本示例的dump_device_info()函數(shù)可以查詢當前設備固件版本信息
以下示例演示通過固件reboot接口重啟設備
// 重啟設備
printf("Reboot device\n");
is_device_removed_ = false;
is_wait_reboot_complete_ = true;
ob_device_reboot(dev, &error);
check_error(error);
// 釋放資源,防止內(nèi)存泄漏
ob_delete_device(dev, &error);
check_error(error);
在ob_device_changed_callback可以監(jiān)聽設備重啟時下線、上線的事件,詳情可以看熱拔插示例
// 監(jiān)聽設備變化
void device_changed_callback(ob_device_list *removed, ob_device_list *added, void *user_data) {
ob_error *error = NULL;
// 通過added處理上線的設備
// 通過removed處理下線的設備
// 釋放資源,避免內(nèi)存泄漏
ob_delete_device_list(removed, &error);
check_error(error);
// 釋放資源,避免內(nèi)存泄漏
ob_delete_device_list(added, &error);
check_error(error);
}
預期輸出:
