blob: db9407b915b2e7b71d41ef0db4d273caf77876a6 [file] [log] [blame]
Delphine CC Chiu22fad392023-10-27 11:05:01 +08001/* SPDX-License-Identifier: Apache-2.0 OR GPL-2.0-or-later */
2#include <libpldm/oem/meta/file_io.h>
3#include <endian.h>
Lora Lin0f5be282024-07-22 09:46:18 +08004#include <stdlib.h>
Delphine CC Chiu22fad392023-10-27 11:05:01 +08005#include <string.h>
6#include <stdio.h>
Delphine CC Chiu22fad392023-10-27 11:05:01 +08007
Lora Lin0f5be282024-07-22 09:46:18 +08008#include "api.h"
9#include "msgbuf.h"
10#include "dsp/base.h"
11
Lora Linea0bf3a2024-09-20 10:09:56 +080012LIBPLDM_ABI_STABLE
Lora Lin0f5be282024-07-22 09:46:18 +080013void *pldm_oem_meta_file_io_write_req_data(
14 struct pldm_oem_meta_file_io_write_req *req)
15{
16 return req->data;
17}
18
Lora Linea0bf3a2024-09-20 10:09:56 +080019LIBPLDM_ABI_STABLE
Lora Lin0f5be282024-07-22 09:46:18 +080020int decode_oem_meta_file_io_write_req(
21 const struct pldm_msg *msg, size_t payload_length,
22 struct pldm_oem_meta_file_io_write_req *req, size_t req_length)
Delphine CC Chiu22fad392023-10-27 11:05:01 +080023{
24 struct pldm_msgbuf _buf;
25 struct pldm_msgbuf *buf = &_buf;
Andrew Jeffery0a1be3c2024-08-11 08:34:10 +000026 int rc;
Delphine CC Chiu22fad392023-10-27 11:05:01 +080027
Lora Lin0f5be282024-07-22 09:46:18 +080028 if (msg == NULL || req == NULL) {
29 return -EINVAL;
Delphine CC Chiu22fad392023-10-27 11:05:01 +080030 }
31
Lora Lin0f5be282024-07-22 09:46:18 +080032 if (req_length < sizeof(*req)) {
33 return -EINVAL;
34 }
35
36 rc = pldm_msgbuf_init_errno(buf,
37 PLDM_OEM_META_FILE_IO_WRITE_REQ_MIN_LENGTH,
38 msg->payload, payload_length);
Delphine CC Chiu22fad392023-10-27 11:05:01 +080039 if (rc) {
40 return rc;
41 }
42
Lora Lin0f5be282024-07-22 09:46:18 +080043 pldm_msgbuf_extract(buf, req->handle);
44 rc = pldm_msgbuf_extract(buf, req->length);
45 if (rc) {
46 return rc;
47 }
Andrew Jeffery0a1be3c2024-08-11 08:34:10 +000048
Lora Lin0f5be282024-07-22 09:46:18 +080049 rc = pldm_msgbuf_extract_array(buf, req->length, req->data,
50 req_length - sizeof(*req));
Andrew Jeffery0a1be3c2024-08-11 08:34:10 +000051 if (rc) {
52 return rc;
53 }
Delphine CC Chiu22fad392023-10-27 11:05:01 +080054
55 return pldm_msgbuf_destroy_consumed(buf);
56}
Lora Lin0f5be282024-07-22 09:46:18 +080057
Andrew Jeffery36324f62024-09-25 13:41:41 +093058LIBPLDM_ABI_DEPRECATED_UNSAFE
Lora Lin0f5be282024-07-22 09:46:18 +080059int decode_oem_meta_file_io_req(const struct pldm_msg *msg,
60 size_t payload_length, uint8_t *file_handle,
61 uint32_t *length, uint8_t *data)
62{
63 struct pldm_oem_meta_file_io_write_req *request_msg;
64 size_t request_msg_len;
65 int rc;
66
67 if (msg == NULL || file_handle == NULL || length == NULL ||
68 data == NULL) {
69 return pldm_xlate_errno(-EINVAL);
70 }
71
Andrew Jeffery36324f62024-09-25 13:41:41 +093072 if (SIZE_MAX - sizeof(*request_msg) < payload_length) {
73 return pldm_xlate_errno(-EOVERFLOW);
74 }
75
Lora Lin0f5be282024-07-22 09:46:18 +080076 request_msg_len = sizeof(*request_msg) + payload_length;
77 request_msg = malloc(request_msg_len);
78
79 rc = decode_oem_meta_file_io_write_req(msg, payload_length, request_msg,
80 request_msg_len);
81 if (rc < 0) {
82 free(request_msg);
83 return pldm_xlate_errno(rc);
84 }
85
86 *file_handle = request_msg->handle;
87 *length = request_msg->length;
88
89 /* NOTE: Unsafe, memory safety is not possible due to API constraints. */
90 memcpy(data, request_msg->data, request_msg->length);
91
92 free(request_msg);
93
94 return 0;
95}
Lora Lin893a08f2024-07-16 15:03:31 +080096
Lora Linea0bf3a2024-09-20 10:09:56 +080097LIBPLDM_ABI_STABLE
Lora Lin893a08f2024-07-16 15:03:31 +080098int decode_oem_meta_file_io_read_req(const struct pldm_msg *msg,
99 size_t payload_length,
100 struct pldm_oem_meta_file_io_read_req *req)
101{
102 struct pldm_msgbuf _buf;
103 struct pldm_msgbuf *buf = &_buf;
104
105 if (msg == NULL || req == NULL) {
106 return -EINVAL;
107 }
108
109 if (req->version > sizeof(struct pldm_oem_meta_file_io_read_req)) {
110 return -E2BIG;
111 }
112
113 int rc = pldm_msgbuf_init_errno(
114 buf, PLDM_OEM_META_FILE_IO_READ_REQ_MIN_LENGTH, msg->payload,
115 payload_length);
116 if (rc) {
117 return rc;
118 }
119
120 pldm_msgbuf_extract(buf, req->handle);
121 rc = pldm_msgbuf_extract(buf, req->option);
122 if (rc) {
123 return rc;
124 }
125
126 rc = pldm_msgbuf_extract(buf, req->length);
127 if (rc) {
128 return rc;
129 }
130
131 switch (req->option) {
132 case PLDM_OEM_META_FILE_IO_READ_ATTR:
133 if (req->length != 0) {
134 return -EPROTO;
135 }
136 break;
137 case PLDM_OEM_META_FILE_IO_READ_DATA:
138 pldm_msgbuf_extract(buf, req->info.data.transferFlag);
139 pldm_msgbuf_extract(buf, req->info.data.offset);
140 break;
141 default:
142 return -EPROTO;
143 }
144
145 return pldm_msgbuf_destroy_consumed(buf);
146}
Lora Lin6476c962024-07-26 14:01:16 +0800147
Lora Linea0bf3a2024-09-20 10:09:56 +0800148LIBPLDM_ABI_STABLE
Lora Lin6476c962024-07-26 14:01:16 +0800149void *pldm_oem_meta_file_io_read_resp_data(
150 struct pldm_oem_meta_file_io_read_resp *resp)
151{
152 return resp->data;
153}
154
Lora Linea0bf3a2024-09-20 10:09:56 +0800155LIBPLDM_ABI_STABLE
Lora Lin6476c962024-07-26 14:01:16 +0800156int encode_oem_meta_file_io_read_resp(
157 uint8_t instance_id, struct pldm_oem_meta_file_io_read_resp *resp,
158 size_t resp_len, struct pldm_msg *responseMsg, size_t payload_length)
159{
160 int rc;
161 struct pldm_msgbuf _buf;
162 struct pldm_msgbuf *buf = &_buf;
163 struct pldm_header_info header = { 0 };
164
165 if (resp == NULL || responseMsg == NULL) {
166 return -EINVAL;
167 }
168
169 if (resp_len < sizeof(*resp)) {
170 return -EINVAL;
171 }
172
173 if (resp->version > sizeof(*resp)) {
174 return -E2BIG;
175 }
176
177 header.instance = instance_id;
178 header.msg_type = PLDM_RESPONSE;
179 header.pldm_type = PLDM_OEM;
180 header.command = PLDM_OEM_META_FILE_IO_CMD_READ_FILE;
181 rc = pack_pldm_header_errno(&header, &(responseMsg->hdr));
182 if (rc) {
183 return rc;
184 }
185
186 rc = pldm_msgbuf_init_errno(buf,
187 PLDM_OEM_META_FILE_IO_READ_RESP_MIN_SIZE,
188 responseMsg->payload, payload_length);
189 if (rc) {
190 return rc;
191 }
192
193 pldm_msgbuf_insert(buf, resp->completion_code);
194 pldm_msgbuf_insert(buf, resp->handle);
195 pldm_msgbuf_insert(buf, resp->option);
196 pldm_msgbuf_insert(buf, resp->length);
197
198 switch (resp->option) {
199 case PLDM_OEM_META_FILE_IO_READ_ATTR:
200 pldm_msgbuf_insert(buf, resp->info.attr.size);
201 pldm_msgbuf_insert(buf, resp->info.attr.crc32);
202 break;
203 case PLDM_OEM_META_FILE_IO_READ_DATA:
204 pldm_msgbuf_insert(buf, resp->info.data.transferFlag);
205 pldm_msgbuf_insert(buf, resp->info.data.offset);
206 rc = pldm_msgbuf_insert_array_uint8(buf, resp->length,
207 resp->data,
208 resp_len - sizeof(*resp));
209 if (rc) {
210 return rc;
211 }
212 break;
213 default:
214 return -EPROTO;
215 }
216
217 return pldm_msgbuf_destroy(buf);
218}