blob: 87284f7bf37f173a3ed8d7d4ea1419466e7e9dac [file] [log] [blame]
Sergii Dmytruk04bd9652023-11-17 19:31:20 +02001/* SPDX-License-Identifier: GPL-2.0-or-later */
2
3#include "storage.h"
4
5#include <assert.h>
6#include <stdio.h>
7
Sergii Dmytruk89e056b2024-03-02 17:02:00 +02008#include "commonlib/bsd/compiler.h"
9#include "fmap.h"
10
Sergii Dmytruk04bd9652023-11-17 19:31:20 +020011#include "fv.h"
12#include "utils.h"
13
14bool storage_open(const char store_file[], struct storage_t *storage, bool rw)
15{
16 storage->rw = rw;
17
18 storage->file = map_file(store_file, rw);
19 if (storage->file.start == NULL) {
20 fprintf(stderr, "Failed to load smm-store-file \"%s\"\n",
21 store_file);
22 return false;
23 }
24
Sergii Dmytruk89e056b2024-03-02 17:02:00 +020025 /* If we won't find FMAP with SMMSTORE, use the whole file, but fail if
26 * FMAP is there without SMMSTORE. */
27 storage->region = storage->file;
28
29 long fmap_offset = fmap_find(storage->file.start, storage->file.length);
30 if (fmap_offset >= 0) {
31 struct fmap *fmap = (void *)(storage->file.start + fmap_offset);
32 const struct fmap_area *area = fmap_find_area(fmap, "SMMSTORE");
33 if (area == NULL) {
34 fprintf(stderr,
35 "Found FMAP without SMMSTORE in \"%s\"\n",
36 store_file);
37 return false;
38 }
39
40 storage->region.start += area->offset;
41 storage->region.length = area->size;
42 }
43
Sergii Dmytruk04bd9652023-11-17 19:31:20 +020044 bool auth_vars;
Sergii Dmytruk89e056b2024-03-02 17:02:00 +020045 if (!fv_parse(storage->region, &storage->store_area, &auth_vars)) {
Sergii Dmytruk04bd9652023-11-17 19:31:20 +020046 if (!rw) {
47 fprintf(stderr,
48 "Failed to find variable store in \"%s\"\n",
49 store_file);
50 goto error;
51 }
52
Maciej Pijanowski9a5cc952024-05-29 13:28:14 +020053 fprintf(stderr,
54 "\nThe variable store has not been found in the ROM image\n"
55 "and is about to be initialized. This situation is normal\n"
56 "for a release image, as the variable store is usually\n"
57 "initialized on the first boot of the platform.\n\n");
58
Sergii Dmytruk89e056b2024-03-02 17:02:00 +020059 if (!fv_init(storage->region)) {
Sergii Dmytruk04bd9652023-11-17 19:31:20 +020060 fprintf(stderr,
61 "Failed to create variable store in \"%s\"\n",
62 store_file);
63 goto error;
64 }
65
Sergii Dmytruk89e056b2024-03-02 17:02:00 +020066 if (!fv_parse(storage->region, &storage->store_area, &auth_vars)) {
Sergii Dmytruk04bd9652023-11-17 19:31:20 +020067 fprintf(stderr,
68 "Failed to parse newly formatted store in \"%s\"\n",
69 store_file);
70 goto error;
71 }
72
73 fprintf(stderr,
74 "Successfully created variable store in \"%s\"\n",
75 store_file);
76 }
77
78 storage->vs = vs_load(storage->store_area, auth_vars);
79 return true;
80
81error:
82 unmap_file(storage->file);
83 return false;
84}
85
86bool storage_write_back(struct storage_t *storage)
87{
88 assert(storage->rw && "Only RW storage can be updated.");
89
90 bool success = vs_store(&storage->vs, storage->store_area);
91 if (!success)
92 fprintf(stderr, "Failed to update variable store\n");
93 storage_drop(storage);
94 return success;
95}
96
97void storage_drop(struct storage_t *storage)
98{
99 unmap_file(storage->file);
100 vs_free(&storage->vs);
101}