Software APIs
boot_svc_next_test.c
1 // Copyright lowRISC contributors (OpenTitan project).
2 // Licensed under the Apache License, Version 2.0, see LICENSE for details.
3 // SPDX-License-Identifier: Apache-2.0
4 
5 #include "sw/device/lib/base/status.h"
8 #include "sw/device/silicon_creator/lib/boot_svc/boot_svc_msg.h"
9 #include "sw/device/silicon_creator/lib/boot_svc/boot_svc_next_boot_bl0_slot.h"
10 #include "sw/device/silicon_creator/lib/drivers/retention_sram.h"
11 #include "sw/device/silicon_creator/lib/drivers/rstmgr.h"
12 #include "sw/device/silicon_creator/rom_ext/e2e/boot_svc/boot_svc_test_lib.h"
13 
14 OTTF_DEFINE_TEST_CONFIG();
15 
16 static status_t initialize(retention_sram_t *retram, boot_svc_retram_t *state) {
17  boot_svc_msg_t msg = {0};
18  boot_svc_next_boot_bl0_slot_req_init(
19  /*primary_slot=*/kBootSlotUnspecified,
20  /*next_slot=*/kBootSlotB, &msg.next_boot_bl0_slot_req);
21  retram->creator.boot_svc_msg = msg;
22  state->state = kBootSvcTestStateNextSideB;
23  rstmgr_reset();
24  return INTERNAL();
25 }
26 
27 static status_t check_side_b(retention_sram_t *retram,
28  boot_svc_retram_t *state) {
29  boot_svc_msg_t msg = retram->creator.boot_svc_msg;
30  TRY(boot_svc_header_check(&msg.header));
31  TRY_CHECK(msg.header.type == kBootSvcNextBl0SlotResType);
32  TRY_CHECK(msg.next_boot_bl0_slot_res.status == kErrorOk);
33  TRY_CHECK(state->current_side == 'B');
34  TRY_CHECK(state->primary_side == 'A');
35  TRY_CHECK(msg.next_boot_bl0_slot_res.primary_bl0_slot == kBootSlotA);
36  state->state = kBootSvcTestStateReturnSideA;
37  rstmgr_reset();
38  return INTERNAL();
39 }
40 
41 static status_t check_return_side_a(retention_sram_t *retram,
42  boot_svc_retram_t *state) {
43  TRY_CHECK(state->current_side == 'A');
44  TRY_CHECK(state->primary_side == 'A');
45  state->state = kBootSvcTestStateFinal;
46  return OK_STATUS();
47 }
48 
49 static status_t next_bl0_slot_test(void) {
50  retention_sram_t *retram = retention_sram_get();
51  TRY(boot_svc_test_init(retram, kBootSvcTestNextBl0));
52  boot_svc_retram_t *state = (boot_svc_retram_t *)&retram->owner;
53 
54  for (;;) {
55  LOG_INFO("Test state = %d", state->state);
56  LOG_INFO("CurrentBootLog: %d:%s", state->boots, state->partition);
57  switch (state->state) {
58  case kBootSvcTestStateInit:
59  TRY(initialize(retram, state));
60  break;
61  case kBootSvcTestStateNextSideB:
62  TRY(check_side_b(retram, state));
63  break;
64  case kBootSvcTestStateReturnSideA:
65  TRY(check_return_side_a(retram, state));
66  break;
67  case kBootSvcTestStateFinal:
68  LOG_INFO("FinalBootLog: %d:%s", state->boots, state->partition);
69  return OK_STATUS();
70  default:
71  LOG_ERROR("Unknown state: %d", state->state);
72  return UNKNOWN();
73  }
74  }
75 }
76 
77 bool test_main(void) {
78  status_t sts = next_bl0_slot_test();
79  if (status_err(sts)) {
80  LOG_ERROR("next_bl0_slot_test: %r", sts);
81  }
82  return status_ok(sts);
83 }