/* * Copyright (C) 2016 Freescale Semiconductor, Inc. * * SPDX-License-Identifier: GPL-2.0+ */ #include #include #include #include #include /* as libavb's bootctl doesn't have the get_var support * we add the getvar support on our side ...*/ #ifndef MAX_PTN #define MAX_PTN 32 #endif #define SLOT_NUM 2 static char *slot_suffix[SLOT_NUM] = {"_a", "_b"}; static int strcmp_l1(const char *s1, const char *s2) { if (!s1 || !s2) return -1; return strncmp(s1, s2, strlen(s1)); } static bool slot_is_bootable(AvbABSlotData* slot) { #ifdef CONFIG_DUAL_BOOTLOADER /* The 'bootloader_verified' will be set when the slot has only one chance * left, which means the slot is bootable even tries_remaining is 0. */ return slot->priority > 0 && (slot->successful_boot || (slot->tries_remaining > 0) || (slot->bootloader_verified == 1)); #else return slot->priority > 0 && (slot->successful_boot || (slot->tries_remaining > 0)); #endif } int slotidx_from_suffix(char *suffix) { int slot = -1; if (!strcmp(suffix, "_a") || !strcmp(suffix, "a")) slot = 0; else if (!strcmp(suffix, "_b") || !strcmp(suffix, "b")) slot = 1; return slot; } bool is_slotvar_avb(char *cmd) { assert(cmd != NULL); if (!strcmp_l1("has-slot:", cmd) || !strcmp_l1("slot-successful:", cmd) || !strcmp_l1("slot-count", cmd) || !strcmp_l1("slot-suffixes", cmd) || !strcmp_l1("current-slot", cmd) || !strcmp_l1("slot-unbootable:", cmd) || !strcmp_l1("slot-retry-count:", cmd)) return true; return false; } int get_curr_slot(AvbABData *ab_data) { if (slot_is_bootable(&ab_data->slots[0]) && slot_is_bootable(&ab_data->slots[1])) { if (ab_data->slots[1].priority > ab_data->slots[0].priority) return 1; else return 0; } else if (slot_is_bootable(&ab_data->slots[0])) return 0; else if (slot_is_bootable(&ab_data->slots[1])) return 1; else return -1; } extern struct fastboot_ptentry g_ptable[MAX_PTN]; extern unsigned int g_pcount; static bool has_slot(char *cmd) { unsigned int n; char *ptr; for (n = 0; n < g_pcount; n++) { ptr = strstr(g_ptable[n].name, cmd); if (ptr != NULL) { ptr += strlen(cmd); if (!strcmp(ptr, "_a") || !strcmp(ptr, "_b")) return true; } } return false; } int get_slotvar_avb(AvbABOps *ab_ops, char *cmd, char *buffer, size_t size) { AvbABData ab_data; AvbABSlotData *slot_data; int slot; if ((ab_ops == NULL) || (cmd == NULL) || (buffer == NULL)) return -1; char *str = cmd; if (!strcmp_l1("has-slot:", cmd)) { str += strlen("has-slot:"); if (has_slot(str)) strlcpy(buffer, "yes", size); else strlcpy(buffer, "no", size); return 0; } else if (!strcmp_l1("slot-suffixes", cmd)) { strlcpy(buffer, "_a,_b", size); return 0 ; } else if (!strcmp_l1("slot-count", cmd)) { strlcpy(buffer, "2", size); return 0 ; } /* load ab meta */ if (ab_ops->read_ab_metadata == NULL || ab_ops->read_ab_metadata(ab_ops, &ab_data) != AVB_IO_RESULT_OK) { strlcpy(buffer, "ab data read error", size); return -1 ; } if (!strcmp_l1("current-slot", cmd)) { int curr = get_curr_slot(&ab_data); if (curr >= 0 && curr < SLOT_NUM) strlcpy(buffer, slot_suffix[curr] + sizeof(unsigned char), size); else { strlcpy(buffer, "no bootable slot", size); return -1; } } else if (!strcmp_l1("slot-successful:", cmd)) { str += strlen("slot-successful:"); slot = slotidx_from_suffix(str); if (slot < 0) { strlcpy(buffer, "no such slot", size); return -1; } else { slot_data = &ab_data.slots[slot]; bool succ = (slot_data->successful_boot != 0); strlcpy(buffer, succ ? "yes" : "no", size); } } else if (!strcmp_l1("slot-unbootable:", cmd)) { str += strlen("slot-unbootable:"); slot = slotidx_from_suffix(str); if (slot < 0) { strlcpy(buffer, "no such slot", size); return -1; } else { slot_data = &ab_data.slots[slot]; bool bootable = slot_is_bootable(slot_data); strlcpy(buffer, bootable ? "no" : "yes", size); } } else if (!strcmp_l1("slot-retry-count:", cmd)) { str += strlen("slot-retry-count:"); slot = slotidx_from_suffix(str); if (slot < 0) { strlcpy(buffer, "no such slot", size); return -1; } else { slot_data = &ab_data.slots[slot]; char var[7]; sprintf(var, "%d", slot_data->tries_remaining); strlcpy(buffer, var, size); } } else { strlcpy(buffer, "no such slot command", size); return -1; } return 0; } char *select_slot(AvbABOps *ab_ops) { AvbABData ab_data; int curr; if (ab_ops == NULL) { return NULL; } /* load ab meta */ if (ab_ops->read_ab_metadata == NULL || ab_ops->read_ab_metadata(ab_ops, &ab_data) != AVB_IO_RESULT_OK) { return NULL; } curr = get_curr_slot(&ab_data); if (curr >= 0 && curr < SLOT_NUM) return slot_suffix[curr]; else return NULL; }