1*8617a60dSAndroid Build Coastguard Worker /* Copyright 2019 The ChromiumOS Authors
2*8617a60dSAndroid Build Coastguard Worker * Use of this source code is governed by a BSD-style license that can be
3*8617a60dSAndroid Build Coastguard Worker * found in the LICENSE file.
4*8617a60dSAndroid Build Coastguard Worker *
5*8617a60dSAndroid Build Coastguard Worker * Functions for initializing the vboot workbuffer and vb2_context.
6*8617a60dSAndroid Build Coastguard Worker */
7*8617a60dSAndroid Build Coastguard Worker
8*8617a60dSAndroid Build Coastguard Worker #include "2api.h"
9*8617a60dSAndroid Build Coastguard Worker #include "2common.h"
10*8617a60dSAndroid Build Coastguard Worker #include "2misc.h"
11*8617a60dSAndroid Build Coastguard Worker #include "2sysincludes.h"
12*8617a60dSAndroid Build Coastguard Worker
vb2_workbuf_from_ctx(struct vb2_context * ctx,struct vb2_workbuf * wb)13*8617a60dSAndroid Build Coastguard Worker void vb2_workbuf_from_ctx(struct vb2_context *ctx, struct vb2_workbuf *wb)
14*8617a60dSAndroid Build Coastguard Worker {
15*8617a60dSAndroid Build Coastguard Worker struct vb2_shared_data *sd = vb2_get_sd(ctx);
16*8617a60dSAndroid Build Coastguard Worker vb2_workbuf_init(wb, (void *)sd + sd->workbuf_used,
17*8617a60dSAndroid Build Coastguard Worker sd->workbuf_size - sd->workbuf_used);
18*8617a60dSAndroid Build Coastguard Worker }
19*8617a60dSAndroid Build Coastguard Worker
vb2_set_workbuf_used(struct vb2_context * ctx,uint32_t used)20*8617a60dSAndroid Build Coastguard Worker void vb2_set_workbuf_used(struct vb2_context *ctx, uint32_t used)
21*8617a60dSAndroid Build Coastguard Worker {
22*8617a60dSAndroid Build Coastguard Worker struct vb2_shared_data *sd = vb2_get_sd(ctx);
23*8617a60dSAndroid Build Coastguard Worker sd->workbuf_used = vb2_wb_round_up(used);
24*8617a60dSAndroid Build Coastguard Worker }
25*8617a60dSAndroid Build Coastguard Worker
vb2api_init(void * workbuf,uint32_t size,struct vb2_context ** ctxptr)26*8617a60dSAndroid Build Coastguard Worker vb2_error_t vb2api_init(void *workbuf, uint32_t size,
27*8617a60dSAndroid Build Coastguard Worker struct vb2_context **ctxptr)
28*8617a60dSAndroid Build Coastguard Worker {
29*8617a60dSAndroid Build Coastguard Worker struct vb2_shared_data *sd = workbuf;
30*8617a60dSAndroid Build Coastguard Worker *ctxptr = NULL;
31*8617a60dSAndroid Build Coastguard Worker
32*8617a60dSAndroid Build Coastguard Worker if (!vb2_aligned(workbuf, VB2_WORKBUF_ALIGN))
33*8617a60dSAndroid Build Coastguard Worker return VB2_ERROR_WORKBUF_ALIGN;
34*8617a60dSAndroid Build Coastguard Worker
35*8617a60dSAndroid Build Coastguard Worker if (size < vb2_wb_round_up(sizeof(*sd)))
36*8617a60dSAndroid Build Coastguard Worker return VB2_ERROR_WORKBUF_SMALL;
37*8617a60dSAndroid Build Coastguard Worker
38*8617a60dSAndroid Build Coastguard Worker /* Zero out vb2_shared_data (which includes vb2_context). */
39*8617a60dSAndroid Build Coastguard Worker memset(sd, 0, sizeof(*sd));
40*8617a60dSAndroid Build Coastguard Worker
41*8617a60dSAndroid Build Coastguard Worker /* Initialize shared data. */
42*8617a60dSAndroid Build Coastguard Worker sd->magic = VB2_SHARED_DATA_MAGIC;
43*8617a60dSAndroid Build Coastguard Worker sd->struct_version_major = VB2_SHARED_DATA_VERSION_MAJOR;
44*8617a60dSAndroid Build Coastguard Worker sd->struct_version_minor = VB2_SHARED_DATA_VERSION_MINOR;
45*8617a60dSAndroid Build Coastguard Worker sd->workbuf_size = size;
46*8617a60dSAndroid Build Coastguard Worker sd->workbuf_used = vb2_wb_round_up(sizeof(*sd));
47*8617a60dSAndroid Build Coastguard Worker
48*8617a60dSAndroid Build Coastguard Worker *ctxptr = &sd->ctx;
49*8617a60dSAndroid Build Coastguard Worker return VB2_SUCCESS;
50*8617a60dSAndroid Build Coastguard Worker }
51*8617a60dSAndroid Build Coastguard Worker
52*8617a60dSAndroid Build Coastguard Worker #pragma GCC diagnostic push
53*8617a60dSAndroid Build Coastguard Worker /* Don't warn for the version_minor check even if the checked version is 0. */
54*8617a60dSAndroid Build Coastguard Worker #pragma GCC diagnostic ignored "-Wtype-limits"
vb2api_relocate(void * new_workbuf,const void * cur_workbuf,uint32_t size,struct vb2_context ** ctxptr)55*8617a60dSAndroid Build Coastguard Worker vb2_error_t vb2api_relocate(void *new_workbuf, const void *cur_workbuf,
56*8617a60dSAndroid Build Coastguard Worker uint32_t size, struct vb2_context **ctxptr)
57*8617a60dSAndroid Build Coastguard Worker {
58*8617a60dSAndroid Build Coastguard Worker const struct vb2_shared_data *cur_sd = cur_workbuf;
59*8617a60dSAndroid Build Coastguard Worker struct vb2_shared_data *new_sd;
60*8617a60dSAndroid Build Coastguard Worker
61*8617a60dSAndroid Build Coastguard Worker if (!vb2_aligned(new_workbuf, VB2_WORKBUF_ALIGN))
62*8617a60dSAndroid Build Coastguard Worker return VB2_ERROR_WORKBUF_ALIGN;
63*8617a60dSAndroid Build Coastguard Worker
64*8617a60dSAndroid Build Coastguard Worker /* Check magic and version. */
65*8617a60dSAndroid Build Coastguard Worker if (cur_sd->magic != VB2_SHARED_DATA_MAGIC)
66*8617a60dSAndroid Build Coastguard Worker return VB2_ERROR_SHARED_DATA_MAGIC;
67*8617a60dSAndroid Build Coastguard Worker
68*8617a60dSAndroid Build Coastguard Worker if (cur_sd->struct_version_major != VB2_SHARED_DATA_VERSION_MAJOR ||
69*8617a60dSAndroid Build Coastguard Worker cur_sd->struct_version_minor < VB2_SHARED_DATA_VERSION_MINOR)
70*8617a60dSAndroid Build Coastguard Worker return VB2_ERROR_SHARED_DATA_VERSION;
71*8617a60dSAndroid Build Coastguard Worker
72*8617a60dSAndroid Build Coastguard Worker /* Check workbuf integrity. */
73*8617a60dSAndroid Build Coastguard Worker if (cur_sd->workbuf_used < vb2_wb_round_up(sizeof(*cur_sd)))
74*8617a60dSAndroid Build Coastguard Worker return VB2_ERROR_WORKBUF_INVALID;
75*8617a60dSAndroid Build Coastguard Worker
76*8617a60dSAndroid Build Coastguard Worker if (cur_sd->workbuf_size < cur_sd->workbuf_used)
77*8617a60dSAndroid Build Coastguard Worker return VB2_ERROR_WORKBUF_INVALID;
78*8617a60dSAndroid Build Coastguard Worker
79*8617a60dSAndroid Build Coastguard Worker if (cur_sd->workbuf_used > size)
80*8617a60dSAndroid Build Coastguard Worker return VB2_ERROR_WORKBUF_SMALL;
81*8617a60dSAndroid Build Coastguard Worker
82*8617a60dSAndroid Build Coastguard Worker /* Relocate if necessary. */
83*8617a60dSAndroid Build Coastguard Worker if (cur_workbuf != new_workbuf)
84*8617a60dSAndroid Build Coastguard Worker memmove(new_workbuf, cur_workbuf, cur_sd->workbuf_used);
85*8617a60dSAndroid Build Coastguard Worker
86*8617a60dSAndroid Build Coastguard Worker /* Set the new size, and return the context pointer. */
87*8617a60dSAndroid Build Coastguard Worker new_sd = new_workbuf;
88*8617a60dSAndroid Build Coastguard Worker new_sd->workbuf_size = size;
89*8617a60dSAndroid Build Coastguard Worker *ctxptr = &new_sd->ctx;
90*8617a60dSAndroid Build Coastguard Worker
91*8617a60dSAndroid Build Coastguard Worker return VB2_SUCCESS;
92*8617a60dSAndroid Build Coastguard Worker }
93*8617a60dSAndroid Build Coastguard Worker #pragma GCC diagnostic pop
94*8617a60dSAndroid Build Coastguard Worker
vb2api_reinit(void * workbuf,struct vb2_context ** ctxptr)95*8617a60dSAndroid Build Coastguard Worker vb2_error_t vb2api_reinit(void *workbuf, struct vb2_context **ctxptr)
96*8617a60dSAndroid Build Coastguard Worker {
97*8617a60dSAndroid Build Coastguard Worker /* Blindly retrieve workbuf_size. vb2api_relocate() will
98*8617a60dSAndroid Build Coastguard Worker perform workbuf validation checks. */
99*8617a60dSAndroid Build Coastguard Worker struct vb2_shared_data *sd = workbuf;
100*8617a60dSAndroid Build Coastguard Worker return vb2api_relocate(workbuf, workbuf, sd->workbuf_size, ctxptr);
101*8617a60dSAndroid Build Coastguard Worker }
102