mirror of https://github.com/F-Stack/f-stack.git
276 lines
7.5 KiB
C
276 lines
7.5 KiB
C
/*
|
|
* BSD LICENSE
|
|
*
|
|
* Copyright(c) 2017 Cavium, Inc.. All rights reserved.
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
*
|
|
* * Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* * Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in
|
|
* the documentation and/or other materials provided with the
|
|
* distribution.
|
|
* * Neither the name of Cavium, Inc. nor the names of its
|
|
* contributors may be used to endorse or promote products derived
|
|
* from this software without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
|
* OWNER(S) OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
#include <rte_ethdev.h>
|
|
#include <rte_cycles.h>
|
|
|
|
#include "lio_logs.h"
|
|
#include "lio_struct.h"
|
|
#include "lio_mbox.h"
|
|
|
|
/**
|
|
* lio_mbox_read:
|
|
* @mbox: Pointer mailbox
|
|
*
|
|
* Reads the 8-bytes of data from the mbox register
|
|
* Writes back the acknowledgment indicating completion of read
|
|
*/
|
|
int
|
|
lio_mbox_read(struct lio_mbox *mbox)
|
|
{
|
|
union lio_mbox_message msg;
|
|
int ret = 0;
|
|
|
|
msg.mbox_msg64 = rte_read64(mbox->mbox_read_reg);
|
|
|
|
if ((msg.mbox_msg64 == LIO_PFVFACK) || (msg.mbox_msg64 == LIO_PFVFSIG))
|
|
return 0;
|
|
|
|
if (mbox->state & LIO_MBOX_STATE_REQ_RECEIVING) {
|
|
mbox->mbox_req.data[mbox->mbox_req.recv_len - 1] =
|
|
msg.mbox_msg64;
|
|
mbox->mbox_req.recv_len++;
|
|
} else {
|
|
if (mbox->state & LIO_MBOX_STATE_RES_RECEIVING) {
|
|
mbox->mbox_resp.data[mbox->mbox_resp.recv_len - 1] =
|
|
msg.mbox_msg64;
|
|
mbox->mbox_resp.recv_len++;
|
|
} else {
|
|
if ((mbox->state & LIO_MBOX_STATE_IDLE) &&
|
|
(msg.s.type == LIO_MBOX_REQUEST)) {
|
|
mbox->state &= ~LIO_MBOX_STATE_IDLE;
|
|
mbox->state |= LIO_MBOX_STATE_REQ_RECEIVING;
|
|
mbox->mbox_req.msg.mbox_msg64 = msg.mbox_msg64;
|
|
mbox->mbox_req.q_no = mbox->q_no;
|
|
mbox->mbox_req.recv_len = 1;
|
|
} else {
|
|
if ((mbox->state &
|
|
LIO_MBOX_STATE_RES_PENDING) &&
|
|
(msg.s.type == LIO_MBOX_RESPONSE)) {
|
|
mbox->state &=
|
|
~LIO_MBOX_STATE_RES_PENDING;
|
|
mbox->state |=
|
|
LIO_MBOX_STATE_RES_RECEIVING;
|
|
mbox->mbox_resp.msg.mbox_msg64 =
|
|
msg.mbox_msg64;
|
|
mbox->mbox_resp.q_no = mbox->q_no;
|
|
mbox->mbox_resp.recv_len = 1;
|
|
} else {
|
|
rte_write64(LIO_PFVFERR,
|
|
mbox->mbox_read_reg);
|
|
mbox->state |= LIO_MBOX_STATE_ERROR;
|
|
return -1;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (mbox->state & LIO_MBOX_STATE_REQ_RECEIVING) {
|
|
if (mbox->mbox_req.recv_len < msg.s.len) {
|
|
ret = 0;
|
|
} else {
|
|
mbox->state &= ~LIO_MBOX_STATE_REQ_RECEIVING;
|
|
mbox->state |= LIO_MBOX_STATE_REQ_RECEIVED;
|
|
ret = 1;
|
|
}
|
|
} else {
|
|
if (mbox->state & LIO_MBOX_STATE_RES_RECEIVING) {
|
|
if (mbox->mbox_resp.recv_len < msg.s.len) {
|
|
ret = 0;
|
|
} else {
|
|
mbox->state &= ~LIO_MBOX_STATE_RES_RECEIVING;
|
|
mbox->state |= LIO_MBOX_STATE_RES_RECEIVED;
|
|
ret = 1;
|
|
}
|
|
} else {
|
|
RTE_ASSERT(0);
|
|
}
|
|
}
|
|
|
|
rte_write64(LIO_PFVFACK, mbox->mbox_read_reg);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* lio_mbox_write:
|
|
* @lio_dev: Pointer lio device
|
|
* @mbox_cmd: Cmd to send to mailbox.
|
|
*
|
|
* Populates the queue specific mbox structure
|
|
* with cmd information.
|
|
* Write the cmd to mbox register
|
|
*/
|
|
int
|
|
lio_mbox_write(struct lio_device *lio_dev,
|
|
struct lio_mbox_cmd *mbox_cmd)
|
|
{
|
|
struct lio_mbox *mbox = lio_dev->mbox[mbox_cmd->q_no];
|
|
uint32_t count, i, ret = LIO_MBOX_STATUS_SUCCESS;
|
|
|
|
if ((mbox_cmd->msg.s.type == LIO_MBOX_RESPONSE) &&
|
|
!(mbox->state & LIO_MBOX_STATE_REQ_RECEIVED))
|
|
return LIO_MBOX_STATUS_FAILED;
|
|
|
|
if ((mbox_cmd->msg.s.type == LIO_MBOX_REQUEST) &&
|
|
!(mbox->state & LIO_MBOX_STATE_IDLE))
|
|
return LIO_MBOX_STATUS_BUSY;
|
|
|
|
if (mbox_cmd->msg.s.type == LIO_MBOX_REQUEST) {
|
|
rte_memcpy(&mbox->mbox_resp, mbox_cmd,
|
|
sizeof(struct lio_mbox_cmd));
|
|
mbox->state = LIO_MBOX_STATE_RES_PENDING;
|
|
}
|
|
|
|
count = 0;
|
|
|
|
while (rte_read64(mbox->mbox_write_reg) != LIO_PFVFSIG) {
|
|
rte_delay_ms(1);
|
|
if (count++ == 1000) {
|
|
ret = LIO_MBOX_STATUS_FAILED;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (ret == LIO_MBOX_STATUS_SUCCESS) {
|
|
rte_write64(mbox_cmd->msg.mbox_msg64, mbox->mbox_write_reg);
|
|
for (i = 0; i < (uint32_t)(mbox_cmd->msg.s.len - 1); i++) {
|
|
count = 0;
|
|
while (rte_read64(mbox->mbox_write_reg) !=
|
|
LIO_PFVFACK) {
|
|
rte_delay_ms(1);
|
|
if (count++ == 1000) {
|
|
ret = LIO_MBOX_STATUS_FAILED;
|
|
break;
|
|
}
|
|
}
|
|
rte_write64(mbox_cmd->data[i], mbox->mbox_write_reg);
|
|
}
|
|
}
|
|
|
|
if (mbox_cmd->msg.s.type == LIO_MBOX_RESPONSE) {
|
|
mbox->state = LIO_MBOX_STATE_IDLE;
|
|
rte_write64(LIO_PFVFSIG, mbox->mbox_read_reg);
|
|
} else {
|
|
if ((!mbox_cmd->msg.s.resp_needed) ||
|
|
(ret == LIO_MBOX_STATUS_FAILED)) {
|
|
mbox->state &= ~LIO_MBOX_STATE_RES_PENDING;
|
|
if (!(mbox->state & (LIO_MBOX_STATE_REQ_RECEIVING |
|
|
LIO_MBOX_STATE_REQ_RECEIVED)))
|
|
mbox->state = LIO_MBOX_STATE_IDLE;
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* lio_mbox_process_cmd:
|
|
* @mbox: Pointer mailbox
|
|
* @mbox_cmd: Pointer to command received
|
|
*
|
|
* Process the cmd received in mbox
|
|
*/
|
|
static int
|
|
lio_mbox_process_cmd(struct lio_mbox *mbox,
|
|
struct lio_mbox_cmd *mbox_cmd)
|
|
{
|
|
struct lio_device *lio_dev = mbox->lio_dev;
|
|
|
|
if (mbox_cmd->msg.s.cmd == LIO_CORES_CRASHED)
|
|
lio_dev_err(lio_dev, "Octeon core(s) crashed or got stuck!\n");
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Process the received mbox message.
|
|
*/
|
|
int
|
|
lio_mbox_process_message(struct lio_mbox *mbox)
|
|
{
|
|
struct lio_mbox_cmd mbox_cmd;
|
|
|
|
if (mbox->state & LIO_MBOX_STATE_ERROR) {
|
|
if (mbox->state & (LIO_MBOX_STATE_RES_PENDING |
|
|
LIO_MBOX_STATE_RES_RECEIVING)) {
|
|
rte_memcpy(&mbox_cmd, &mbox->mbox_resp,
|
|
sizeof(struct lio_mbox_cmd));
|
|
mbox->state = LIO_MBOX_STATE_IDLE;
|
|
rte_write64(LIO_PFVFSIG, mbox->mbox_read_reg);
|
|
mbox_cmd.recv_status = 1;
|
|
if (mbox_cmd.fn)
|
|
mbox_cmd.fn(mbox->lio_dev, &mbox_cmd,
|
|
mbox_cmd.fn_arg);
|
|
|
|
return 0;
|
|
}
|
|
|
|
mbox->state = LIO_MBOX_STATE_IDLE;
|
|
rte_write64(LIO_PFVFSIG, mbox->mbox_read_reg);
|
|
|
|
return 0;
|
|
}
|
|
|
|
if (mbox->state & LIO_MBOX_STATE_RES_RECEIVED) {
|
|
rte_memcpy(&mbox_cmd, &mbox->mbox_resp,
|
|
sizeof(struct lio_mbox_cmd));
|
|
mbox->state = LIO_MBOX_STATE_IDLE;
|
|
rte_write64(LIO_PFVFSIG, mbox->mbox_read_reg);
|
|
mbox_cmd.recv_status = 0;
|
|
if (mbox_cmd.fn)
|
|
mbox_cmd.fn(mbox->lio_dev, &mbox_cmd, mbox_cmd.fn_arg);
|
|
|
|
return 0;
|
|
}
|
|
|
|
if (mbox->state & LIO_MBOX_STATE_REQ_RECEIVED) {
|
|
rte_memcpy(&mbox_cmd, &mbox->mbox_req,
|
|
sizeof(struct lio_mbox_cmd));
|
|
if (!mbox_cmd.msg.s.resp_needed) {
|
|
mbox->state &= ~LIO_MBOX_STATE_REQ_RECEIVED;
|
|
if (!(mbox->state & LIO_MBOX_STATE_RES_PENDING))
|
|
mbox->state = LIO_MBOX_STATE_IDLE;
|
|
rte_write64(LIO_PFVFSIG, mbox->mbox_read_reg);
|
|
}
|
|
|
|
lio_mbox_process_cmd(mbox, &mbox_cmd);
|
|
|
|
return 0;
|
|
}
|
|
|
|
RTE_ASSERT(0);
|
|
|
|
return 0;
|
|
}
|