//
// Copyright 2014 Ettus Research LLC
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program. If not, see .
//
#include
#include
#include //memcmp
bool process_fw_comm_protocol_pkt(
const fw_comm_pkt_t* request,
fw_comm_pkt_t* response,
uint8_t product_id,
uint32_t iface_id,
poke32_func poke_callback,
peek32_func peek_callback)
{
bool send_response = false;
uint16_t signature = request->id;
uint8_t version = FW_COMM_GET_PROTOCOL_VER(request->id);
uint8_t product = FW_COMM_GET_PRODUCT_ID(request->id);
if (signature == FW_COMM_PROTOCOL_SIGNATURE && //Verify protocol
version <= FW_COMM_PROTOCOL_VERSION && //Verify protocol version (older versions supported)
product == product_id) //Verify device
{
//Request is valid. Copy it into the reply.
memcpy(response, request, sizeof(fw_comm_pkt_t));
//Start assuming no error
response->flags &= ~FW_COMM_FLAGS_ERROR_MASK;
//Otherwise, run the command set by the flags
switch (request->flags & FW_COMM_FLAGS_CMD_MASK) {
case FW_COMM_CMD_ECHO: {
UHD_FW_TRACE(DEBUG, "fw_comm_protocol::echo()");
response->data_words = 1;
response->data[0] = iface_id;
} break;
case FW_COMM_CMD_POKE32: {
UHD_FW_TRACE_FSTR(DEBUG, "fw_comm_protocol::poke32(0x%x)=0x%x",
request->addr,*(request->data));
poke_callback(request->addr, *(request->data));
} break;
case FW_COMM_CMD_PEEK32: {
*(response->data) = peek_callback(request->addr);
UHD_FW_TRACE_FSTR(DEBUG, "fw_comm_protocol::peek32(0x%x)=0x%x",
request->addr,*(response->data));
} break;
case FW_COMM_CMD_BLOCK_POKE32: {
if (request->data_words > FW_COMM_MAX_DATA_WORDS) {
response->flags |= FW_COMM_ERR_SIZE_ERROR;
response->data_words = FW_COMM_MAX_DATA_WORDS;
} else {
response->data_words = request->data_words;
}
UHD_FW_TRACE_FSTR(DEBUG, "fw_comm_protocol::block_poke32(0x%x,%d)",request->addr,response->data_words);
for (uint32_t i = 0; i < response->data_words; i++) {
poke_callback(request->addr + (i * sizeof(uint32_t)), request->data[i]);
}
} break;
case FW_COMM_CMD_BLOCK_PEEK32: {
if (request->data_words > FW_COMM_MAX_DATA_WORDS) {
response->flags |= FW_COMM_ERR_SIZE_ERROR;
response->data_words = FW_COMM_MAX_DATA_WORDS;
} else {
response->data_words = request->data_words;
}
for (uint32_t i = 0; i < response->data_words; i++) {
response->data[i] = peek_callback(request->addr + (i * sizeof(uint32_t)));
}
UHD_FW_TRACE_FSTR(DEBUG, "fw_comm_protocol::block_peek32(0x%x,%d)",request->addr,response->data_words);
} break;
default: {
UHD_FW_TRACE(ERROR, "fw_comm_protocol got an invalid command.");
response->flags |= FW_COMM_ERR_CMD_ERROR;
}
}
//Send a reply if ack requested
send_response = (request->flags & FW_COMM_FLAGS_ACK);
} else { //Size, protocol, product check failed
UHD_FW_TRACE(WARN, "fw_comm_protocol ignored an unknown request.");
send_response = false;
}
return send_response;
}