LCOV - code coverage report
Current view: top level - library/spdm_responder_lib - libspdm_rsp_vendor_response.c (source / functions) Coverage Total Hit
Test: coverage.info Lines: 93.4 % 106 99
Test Date: 2025-09-14 08:11:04 Functions: 100.0 % 3 3

            Line data    Source code
       1              : /**
       2              :  *  Copyright Notice:
       3              :  *  Copyright 2023-2025 DMTF. All rights reserved.
       4              :  *  License: BSD 3-Clause License. For full text see link: https://github.com/DMTF/libspdm/blob/main/LICENSE.md
       5              :  **/
       6              : 
       7              : #include "internal/libspdm_responder_lib.h"
       8              : 
       9              : #if LIBSPDM_ENABLE_VENDOR_DEFINED_MESSAGES
      10              : 
      11              : /* expected number of bytes for VENDOR MESSAGE HEADERS */
      12              : #define SPDM_VENDOR_DEFINED_FIXED_HEADER_LEN 7
      13              : 
      14            9 : libspdm_return_t libspdm_register_vendor_get_id_callback_func(void *spdm_context,
      15              :                                                               libspdm_vendor_get_id_callback_func resp_callback)
      16              : {
      17              : 
      18            9 :     libspdm_context_t *context = (libspdm_context_t *)spdm_context;
      19            9 :     context->vendor_response_get_id = resp_callback;
      20            9 :     return LIBSPDM_STATUS_SUCCESS;
      21              : }
      22              : 
      23           10 : libspdm_return_t libspdm_register_vendor_callback_func(void *spdm_context,
      24              :                                                        libspdm_vendor_response_callback_func resp_callback)
      25              : {
      26              : 
      27           10 :     libspdm_context_t *context = (libspdm_context_t *)spdm_context;
      28           10 :     context->vendor_response_callback = resp_callback;
      29           10 :     return LIBSPDM_STATUS_SUCCESS;
      30              : }
      31              : 
      32           11 : libspdm_return_t libspdm_get_vendor_defined_response(libspdm_context_t *spdm_context,
      33              :                                                      size_t request_size,
      34              :                                                      const void *request,
      35              :                                                      size_t *response_size,
      36              :                                                      void *response)
      37              : {
      38              :     const spdm_vendor_defined_request_msg_t *spdm_request;
      39              :     spdm_vendor_defined_response_msg_t *spdm_response;
      40              :     uint16_t header_length;
      41           11 :     size_t response_capacity = 0;
      42           11 :     libspdm_return_t status = LIBSPDM_STATUS_SUCCESS;
      43              : 
      44           11 :     libspdm_session_info_t *session_info = NULL;
      45           11 :     libspdm_session_state_t session_state = 0;
      46           11 :     const uint32_t *session_id = NULL;
      47           11 :     uint8_t *resp_data = NULL;
      48              :     const uint8_t *req_vendor_id;
      49              :     const uint8_t *req_data;
      50           11 :     uint32_t resp_size = 0;
      51           11 :     uint32_t req_size = 0;
      52              :     bool use_large_payload;
      53              : 
      54              :     /* -=[Check Parameters Phase]=- */
      55           11 :     if (request == NULL ||
      56           10 :         response == NULL ||
      57              :         response_size == NULL) {
      58            1 :         return LIBSPDM_STATUS_INVALID_PARAMETER;
      59              :     }
      60              : 
      61           10 :     if (spdm_context->last_spdm_request_session_id_valid) {
      62            2 :         session_info = libspdm_get_session_info_via_session_id(
      63              :             spdm_context,
      64              :             spdm_context->last_spdm_request_session_id);
      65            2 :         if (session_info == NULL) {
      66            0 :             return libspdm_generate_error_response(
      67              :                 spdm_context,
      68              :                 SPDM_ERROR_CODE_UNEXPECTED_REQUEST, 0,
      69              :                 response_size, response);
      70              :         }
      71            2 :         session_state = libspdm_secured_message_get_session_state(
      72              :             session_info->secured_message_context);
      73            2 :         if (session_state != LIBSPDM_SESSION_STATE_ESTABLISHED) {
      74            0 :             return libspdm_generate_error_response(
      75              :                 spdm_context,
      76              :                 SPDM_ERROR_CODE_UNEXPECTED_REQUEST, 0,
      77              :                 response_size, response);
      78              :         }
      79            2 :         session_id = &session_info->session_id;
      80              :     }
      81              : 
      82              :     /* Check if caller is using the old Vendor Defined API. */
      83           10 :     if ((spdm_context->vendor_response_callback == NULL ||
      84            8 :          spdm_context->vendor_response_get_id == NULL)) {
      85            2 :         if (spdm_context->get_response_func != NULL) {
      86            1 :             return ((libspdm_get_response_func)spdm_context->get_response_func)(
      87              :                 spdm_context,
      88              :                 session_id,
      89              :                 false,
      90              :                 request_size,
      91              :                 request,
      92              :                 response_size,
      93              :                 response);
      94              :         } else
      95            1 :             return libspdm_generate_error_response(spdm_context,
      96              :                                                    SPDM_ERROR_CODE_UNSUPPORTED_REQUEST,
      97              :                                                    SPDM_VENDOR_DEFINED_REQUEST,
      98              :                                                    response_size, response);
      99              :     }
     100              : 
     101            8 :     spdm_request = request;
     102              : 
     103            8 :     if (spdm_request->header.spdm_version != libspdm_get_connection_version(spdm_context)) {
     104            0 :         return libspdm_generate_error_response(spdm_context,
     105              :                                                SPDM_ERROR_CODE_VERSION_MISMATCH, 0,
     106              :                                                response_size, response);
     107              :     }
     108            8 :     if (spdm_context->response_state != LIBSPDM_RESPONSE_STATE_NORMAL) {
     109            0 :         return libspdm_responder_handle_response_state(
     110              :             spdm_context,
     111            0 :             spdm_request->header.request_response_code,
     112              :             response_size, response);
     113              :     }
     114            8 :     if (spdm_context->connection_info.connection_state < LIBSPDM_CONNECTION_STATE_NEGOTIATED) {
     115            0 :         return libspdm_generate_error_response(spdm_context,
     116              :                                                SPDM_ERROR_CODE_UNEXPECTED_REQUEST,
     117              :                                                0, response_size, response);
     118              :     }
     119              : 
     120            8 :     if ((spdm_request->header.spdm_version >= SPDM_MESSAGE_VERSION_14) &&
     121            4 :         ((spdm_request->header.param1 & SPDM_VENDOR_DEFINED_REQUEST_LARGE_REQ) != 0)) {
     122            4 :         if (!libspdm_is_capabilities_flag_supported(
     123              :                 spdm_context, false,
     124              :                 SPDM_GET_CAPABILITIES_REQUEST_FLAGS_LARGE_RESP_CAP,
     125              :                 SPDM_GET_CAPABILITIES_RESPONSE_FLAGS_LARGE_RESP_CAP)) {
     126            1 :             return libspdm_generate_error_response(
     127              :                 spdm_context, SPDM_ERROR_CODE_INVALID_REQUEST,
     128              :                 0, response_size, response);
     129              :         }
     130              : 
     131            3 :         use_large_payload = true;
     132              :     } else {
     133            4 :         use_large_payload = false;
     134              :     }
     135              : 
     136              : 
     137            7 :     if (request_size < sizeof(spdm_vendor_defined_request_msg_t)) {
     138            0 :         return libspdm_generate_error_response(spdm_context,
     139              :                                                SPDM_ERROR_CODE_INVALID_REQUEST, 0,
     140              :                                                response_size, response);
     141              :     }
     142            7 :     if (use_large_payload) {
     143            3 :         if (request_size < sizeof(spdm_vendor_defined_request_msg_t) +
     144            3 :             spdm_request->len + sizeof(uint16_t) + sizeof(uint32_t)) {
     145            1 :             return libspdm_generate_error_response(spdm_context,
     146              :                                                    SPDM_ERROR_CODE_INVALID_REQUEST, 0,
     147              :                                                    response_size, response);
     148              :         }
     149              :     } else {
     150            4 :         if (request_size < sizeof(spdm_vendor_defined_request_msg_t) +
     151            4 :             spdm_request->len + sizeof(uint16_t)) {
     152            2 :             return libspdm_generate_error_response(spdm_context,
     153              :                                                    SPDM_ERROR_CODE_INVALID_REQUEST, 0,
     154              :                                                    response_size, response);
     155              :         }
     156              :     }
     157            4 :     req_vendor_id = ((const uint8_t *)request) + sizeof(spdm_vendor_defined_request_msg_t);
     158              : 
     159            4 :     if (use_large_payload) {
     160            2 :         req_size = libspdm_read_uint32((const uint8_t *)(req_vendor_id + spdm_request->len + sizeof(uint16_t)));
     161            2 :         if (request_size < sizeof(spdm_vendor_defined_request_msg_t) +
     162            2 :             spdm_request->len + sizeof(uint16_t) + sizeof(uint32_t) + req_size) {
     163            1 :             return libspdm_generate_error_response(spdm_context,
     164              :                                                    SPDM_ERROR_CODE_INVALID_REQUEST, 0,
     165              :                                                    response_size, response);
     166              :         }
     167              :     } else {
     168            2 :         req_size = libspdm_read_uint16((const uint8_t *)(req_vendor_id + spdm_request->len));
     169            2 :         if (request_size < sizeof(spdm_vendor_defined_request_msg_t) +
     170            2 :             spdm_request->len + sizeof(uint16_t) + req_size) {
     171            1 :             return libspdm_generate_error_response(spdm_context,
     172              :                                                    SPDM_ERROR_CODE_INVALID_REQUEST, 0,
     173              :                                                    response_size, response);
     174              :         }
     175              :     }
     176              : 
     177            2 :     libspdm_reset_message_buffer_via_request_code(spdm_context, NULL,
     178            2 :                                                   spdm_request->header.request_response_code);
     179              : 
     180              :     /* length of spdm request/response header before payload start */
     181            2 :     header_length = sizeof(spdm_vendor_defined_response_msg_t) + spdm_request->len +
     182              :                     sizeof(uint16_t);
     183            2 :     if (use_large_payload) {
     184            1 :         header_length += sizeof(uint32_t);
     185              :     }
     186              : 
     187            2 :     LIBSPDM_ASSERT(*response_size >= header_length);
     188              :     LIBSPDM_ASSERT(
     189              :         sizeof(spdm_vendor_defined_response_msg_t) == SPDM_VENDOR_DEFINED_FIXED_HEADER_LEN);
     190            2 :     response_capacity = *response_size - header_length;
     191            2 :     libspdm_zero_mem(response, header_length);
     192            2 :     spdm_response = response;
     193              : 
     194            2 :     spdm_response->header.spdm_version = spdm_request->header.spdm_version;
     195            2 :     spdm_response->header.request_response_code = SPDM_VENDOR_DEFINED_RESPONSE;
     196            2 :     spdm_response->header.param1 = 0;
     197            2 :     spdm_response->header.param2 = 0;
     198              : 
     199            2 :     if (use_large_payload) {
     200            1 :         spdm_response->header.param1 |= SPDM_VENDOR_DEFINED_RESONSE_LARGE_RESP;
     201              :     }
     202              : 
     203              :     /* SPDM Response format
     204              :      *  1 byte SPDMVersion
     205              :      *  1 byte RequestResponseCode
     206              :      *  2 bytes Reserved
     207              :      *  2 bytes StandardID
     208              :      *  1 bytes VendorID Length Len1, based on StandardID
     209              :      *  Len1 bytes VendorID
     210              :      *  if LargeResp == 1
     211              :      *      2 bytes Reserved
     212              :      *      4 bytes Response Length Len2
     213              :      *      Len2 bytes Response Payload
     214              :      *  if LargeResp == 0
     215              :      *      2 bytes Response Length Len2
     216              :      *      Len2 bytes Response Payload
     217              :      */
     218              : 
     219              :     /* replace capacity with size */
     220            2 :     spdm_response->len = SPDM_MAX_VENDOR_ID_LENGTH;
     221            2 :     resp_data = ((uint8_t *)response) + sizeof(spdm_vendor_defined_response_msg_t);
     222              : 
     223            2 :     if (use_large_payload) {
     224            1 :         req_data = ((const uint8_t *)request) +
     225              :                    sizeof(spdm_vendor_defined_request_msg_t) +
     226            1 :                    ((const spdm_vendor_defined_request_msg_t*)request)->len +
     227            1 :                    sizeof(uint16_t) + sizeof(uint32_t);
     228              :     } else {
     229            1 :         req_data = ((const uint8_t *)request) +
     230              :                    sizeof(spdm_vendor_defined_request_msg_t) +
     231            1 :                    ((const spdm_vendor_defined_request_msg_t*)request)->len +
     232              :                    sizeof(uint16_t);
     233              :     }
     234              : 
     235            2 :     status = spdm_context->vendor_response_get_id(
     236              :         spdm_context,
     237              :         session_id,
     238              :         &spdm_response->standard_id,
     239              :         &spdm_response->len,
     240              :         resp_data);
     241              : 
     242              :     /* move pointer and adjust buffer size */
     243            2 :     if (use_large_payload) {
     244            1 :         resp_data += spdm_response->len + sizeof(uint16_t) + sizeof(uint32_t);
     245            1 :         response_capacity -= spdm_response->len + sizeof(uint16_t) + sizeof(uint32_t);
     246            1 :         resp_size = (uint32_t)response_capacity;
     247              :     } else {
     248            1 :         resp_data += spdm_response->len + sizeof(uint16_t);
     249            1 :         response_capacity -= spdm_response->len + sizeof(uint16_t);
     250            1 :         resp_size = (uint16_t)response_capacity;
     251              :     }
     252              : 
     253            2 :     status = spdm_context->vendor_response_callback(spdm_context,
     254              :                                                     session_id,
     255            2 :                                                     spdm_request->standard_id,
     256            2 :                                                     spdm_request->len,
     257              :                                                     req_vendor_id, req_size, req_data,
     258              :                                                     &resp_size,
     259              :                                                     resp_data);
     260              : 
     261              :     /* store back the response payload size */
     262            2 :     if (use_large_payload) {
     263            1 :         libspdm_write_uint32((uint8_t *)(resp_data - sizeof(uint32_t)), resp_size);
     264            1 :         *response_size = resp_size + (size_t)header_length;
     265              :     } else {
     266            1 :         libspdm_write_uint16((uint8_t *)(resp_data - sizeof(uint16_t)), (uint16_t)resp_size);
     267            1 :         *response_size = (uint16_t)resp_size + (size_t)header_length;
     268              :     }
     269              : 
     270              :     LIBSPDM_ASSERT(sizeof(spdm_vendor_defined_request_msg_t) ==
     271              :                    SPDM_VENDOR_DEFINED_FIXED_HEADER_LEN);
     272              : 
     273            2 :     return status;
     274              : }
     275              : 
     276              : #endif /* LIBSPDM_ENABLE_VENDOR_DEFINED_MESSAGES */
        

Generated by: LCOV version 2.0-1