554 lines
12 KiB
C
554 lines
12 KiB
C
/*-
|
|
* Copyright (c) 2011-2012 Semihalf.
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. 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.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR 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 AUTHOR 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 <sys/cdefs.h>
|
|
__FBSDID("$FreeBSD$");
|
|
|
|
#include <sys/param.h>
|
|
#include <sys/systm.h>
|
|
#include <sys/kernel.h>
|
|
#include <sys/bus.h>
|
|
#include <sys/lock.h>
|
|
#include <sys/module.h>
|
|
#include <sys/mutex.h>
|
|
#include <sys/proc.h>
|
|
#include <sys/pcpu.h>
|
|
#include <sys/rman.h>
|
|
#include <sys/sched.h>
|
|
#include <sys/smp.h>
|
|
|
|
#include <machine/bus.h>
|
|
#include <machine/resource.h>
|
|
#include <machine/tlb.h>
|
|
|
|
#include "qman.h"
|
|
#include "portals.h"
|
|
|
|
extern struct dpaa_portals_softc *qp_sc;
|
|
static struct qman_softc *qman_sc;
|
|
|
|
extern t_Handle qman_portal_setup(struct qman_softc *qsc);
|
|
|
|
static void
|
|
qman_exception(t_Handle app, e_QmExceptions exception)
|
|
{
|
|
struct qman_softc *sc;
|
|
const char *message;
|
|
|
|
sc = app;
|
|
|
|
switch (exception) {
|
|
case e_QM_EX_CORENET_INITIATOR_DATA:
|
|
message = "Initiator Data Error";
|
|
break;
|
|
case e_QM_EX_CORENET_TARGET_DATA:
|
|
message = "CoreNet Target Data Error";
|
|
break;
|
|
case e_QM_EX_CORENET_INVALID_TARGET_TRANSACTION:
|
|
message = "Invalid Target Transaction";
|
|
break;
|
|
case e_QM_EX_PFDR_THRESHOLD:
|
|
message = "PFDR Low Watermark Interrupt";
|
|
break;
|
|
case e_QM_EX_PFDR_ENQUEUE_BLOCKED:
|
|
message = "PFDR Enqueues Blocked Interrupt";
|
|
break;
|
|
case e_QM_EX_SINGLE_ECC:
|
|
message = "Single Bit ECC Error Interrupt";
|
|
break;
|
|
case e_QM_EX_MULTI_ECC:
|
|
message = "Multi Bit ECC Error Interrupt";
|
|
break;
|
|
case e_QM_EX_INVALID_COMMAND:
|
|
message = "Invalid Command Verb Interrupt";
|
|
break;
|
|
case e_QM_EX_DEQUEUE_DCP:
|
|
message = "Invalid Dequeue Direct Connect Portal Interrupt";
|
|
break;
|
|
case e_QM_EX_DEQUEUE_FQ:
|
|
message = "Invalid Dequeue FQ Interrupt";
|
|
break;
|
|
case e_QM_EX_DEQUEUE_SOURCE:
|
|
message = "Invalid Dequeue Source Interrupt";
|
|
break;
|
|
case e_QM_EX_DEQUEUE_QUEUE:
|
|
message = "Invalid Dequeue Queue Interrupt";
|
|
break;
|
|
case e_QM_EX_ENQUEUE_OVERFLOW:
|
|
message = "Invalid Enqueue Overflow Interrupt";
|
|
break;
|
|
case e_QM_EX_ENQUEUE_STATE:
|
|
message = "Invalid Enqueue State Interrupt";
|
|
break;
|
|
case e_QM_EX_ENQUEUE_CHANNEL:
|
|
message = "Invalid Enqueue Channel Interrupt";
|
|
break;
|
|
case e_QM_EX_ENQUEUE_QUEUE:
|
|
message = "Invalid Enqueue Queue Interrupt";
|
|
break;
|
|
case e_QM_EX_CG_STATE_CHANGE:
|
|
message = "CG change state notification";
|
|
break;
|
|
default:
|
|
message = "Unknown error";
|
|
}
|
|
|
|
device_printf(sc->sc_dev, "QMan Exception: %s.\n", message);
|
|
}
|
|
|
|
/**
|
|
* General received frame callback.
|
|
* This is called, when user did not register his own callback for a given
|
|
* frame queue range (fqr).
|
|
*/
|
|
e_RxStoreResponse
|
|
qman_received_frame_callback(t_Handle app, t_Handle qm_fqr, t_Handle qm_portal,
|
|
uint32_t fqid_offset, t_DpaaFD *frame)
|
|
{
|
|
struct qman_softc *sc;
|
|
|
|
sc = app;
|
|
|
|
device_printf(sc->sc_dev, "dummy callback for received frame.\n");
|
|
return (e_RX_STORE_RESPONSE_CONTINUE);
|
|
}
|
|
|
|
/**
|
|
* General rejected frame callback.
|
|
* This is called, when user did not register his own callback for a given
|
|
* frame queue range (fqr).
|
|
*/
|
|
e_RxStoreResponse
|
|
qman_rejected_frame_callback(t_Handle app, t_Handle qm_fqr, t_Handle qm_portal,
|
|
uint32_t fqid_offset, t_DpaaFD *frame,
|
|
t_QmRejectedFrameInfo *qm_rejected_frame_info)
|
|
{
|
|
struct qman_softc *sc;
|
|
|
|
sc = app;
|
|
|
|
device_printf(sc->sc_dev, "dummy callback for rejected frame.\n");
|
|
return (e_RX_STORE_RESPONSE_CONTINUE);
|
|
}
|
|
|
|
int
|
|
qman_attach(device_t dev)
|
|
{
|
|
struct qman_softc *sc;
|
|
t_QmParam qp;
|
|
t_Error error;
|
|
t_QmRevisionInfo rev;
|
|
|
|
sc = device_get_softc(dev);
|
|
sc->sc_dev = dev;
|
|
qman_sc = sc;
|
|
|
|
if (XX_MallocSmartInit() != E_OK) {
|
|
device_printf(dev, "could not initialize smart allocator.\n");
|
|
return (ENXIO);
|
|
}
|
|
|
|
sched_pin();
|
|
|
|
/* Allocate resources */
|
|
sc->sc_rrid = 0;
|
|
sc->sc_rres = bus_alloc_resource(dev, SYS_RES_MEMORY,
|
|
&sc->sc_rrid, 0, ~0, QMAN_CCSR_SIZE, RF_ACTIVE);
|
|
if (sc->sc_rres == NULL) {
|
|
device_printf(dev, "could not allocate memory.\n");
|
|
goto err;
|
|
}
|
|
|
|
sc->sc_irid = 0;
|
|
sc->sc_ires = bus_alloc_resource_any(dev, SYS_RES_IRQ,
|
|
&sc->sc_irid, RF_ACTIVE | RF_SHAREABLE);
|
|
if (sc->sc_ires == NULL) {
|
|
device_printf(dev, "could not allocate error interrupt.\n");
|
|
goto err;
|
|
}
|
|
|
|
if (qp_sc == NULL)
|
|
goto err;
|
|
|
|
dpaa_portal_map_registers(qp_sc);
|
|
|
|
/* Initialize QMan */
|
|
qp.guestId = NCSW_MASTER_ID;
|
|
qp.baseAddress = rman_get_bushandle(sc->sc_rres);
|
|
qp.swPortalsBaseAddress = rman_get_bushandle(qp_sc->sc_rres[0]);
|
|
qp.liodn = 0;
|
|
qp.totalNumOfFqids = QMAN_MAX_FQIDS;
|
|
qp.fqdMemPartitionId = NCSW_MASTER_ID;
|
|
qp.pfdrMemPartitionId = NCSW_MASTER_ID;
|
|
qp.f_Exception = qman_exception;
|
|
qp.h_App = sc;
|
|
qp.errIrq = (uintptr_t)sc->sc_ires;
|
|
qp.partFqidBase = QMAN_FQID_BASE;
|
|
qp.partNumOfFqids = QMAN_MAX_FQIDS;
|
|
qp.partCgsBase = 0;
|
|
qp.partNumOfCgs = 0;
|
|
|
|
sc->sc_qh = QM_Config(&qp);
|
|
if (sc->sc_qh == NULL) {
|
|
device_printf(dev, "could not be configured\n");
|
|
goto err;
|
|
}
|
|
|
|
error = QM_Init(sc->sc_qh);
|
|
if (error != E_OK) {
|
|
device_printf(dev, "could not be initialized\n");
|
|
goto err;
|
|
}
|
|
|
|
error = QM_GetRevision(sc->sc_qh, &rev);
|
|
if (error != E_OK) {
|
|
device_printf(dev, "could not get QMan revision\n");
|
|
goto err;
|
|
}
|
|
|
|
device_printf(dev, "Hardware version: %d.%d.\n",
|
|
rev.majorRev, rev.minorRev);
|
|
|
|
sched_unpin();
|
|
|
|
qman_portal_setup(sc);
|
|
|
|
return (0);
|
|
|
|
err:
|
|
sched_unpin();
|
|
qman_detach(dev);
|
|
return (ENXIO);
|
|
}
|
|
|
|
int
|
|
qman_detach(device_t dev)
|
|
{
|
|
struct qman_softc *sc;
|
|
|
|
sc = device_get_softc(dev);
|
|
|
|
if (sc->sc_qh)
|
|
QM_Free(sc->sc_qh);
|
|
|
|
if (sc->sc_ires != NULL)
|
|
XX_DeallocIntr((uintptr_t)sc->sc_ires);
|
|
|
|
if (sc->sc_ires != NULL)
|
|
bus_release_resource(dev, SYS_RES_IRQ,
|
|
sc->sc_irid, sc->sc_ires);
|
|
|
|
if (sc->sc_rres != NULL)
|
|
bus_release_resource(dev, SYS_RES_MEMORY,
|
|
sc->sc_rrid, sc->sc_rres);
|
|
|
|
return (0);
|
|
}
|
|
|
|
int
|
|
qman_suspend(device_t dev)
|
|
{
|
|
|
|
return (0);
|
|
}
|
|
|
|
int
|
|
qman_resume(device_t dev)
|
|
{
|
|
|
|
return (0);
|
|
}
|
|
|
|
int
|
|
qman_shutdown(device_t dev)
|
|
{
|
|
|
|
return (0);
|
|
}
|
|
|
|
|
|
/**
|
|
* @group QMan API functions implementation.
|
|
* @{
|
|
*/
|
|
|
|
t_Handle
|
|
qman_fqr_create(uint32_t fqids_num, e_QmFQChannel channel, uint8_t wq,
|
|
bool force_fqid, uint32_t fqid_or_align, bool init_parked,
|
|
bool hold_active, bool prefer_in_cache, bool congst_avoid_ena,
|
|
t_Handle congst_group, int8_t overhead_accounting_len,
|
|
uint32_t tail_drop_threshold)
|
|
{
|
|
struct qman_softc *sc;
|
|
t_QmFqrParams fqr;
|
|
t_Handle fqrh, portal;
|
|
|
|
sc = qman_sc;
|
|
|
|
sched_pin();
|
|
|
|
/* Ensure we have got QMan port initialized */
|
|
portal = qman_portal_setup(sc);
|
|
if (portal == NULL) {
|
|
device_printf(sc->sc_dev, "could not setup QMan portal\n");
|
|
goto err;
|
|
}
|
|
|
|
fqr.h_Qm = sc->sc_qh;
|
|
fqr.h_QmPortal = portal;
|
|
fqr.initParked = init_parked;
|
|
fqr.holdActive = hold_active;
|
|
fqr.preferInCache = prefer_in_cache;
|
|
|
|
/* We do not support stashing */
|
|
fqr.useContextAForStash = FALSE;
|
|
fqr.p_ContextA = 0;
|
|
fqr.p_ContextB = 0;
|
|
|
|
fqr.channel = channel;
|
|
fqr.wq = wq;
|
|
fqr.shadowMode = FALSE;
|
|
fqr.numOfFqids = fqids_num;
|
|
|
|
/* FQID */
|
|
fqr.useForce = force_fqid;
|
|
if (force_fqid) {
|
|
fqr.qs.frcQ.fqid = fqid_or_align;
|
|
} else {
|
|
fqr.qs.nonFrcQs.align = fqid_or_align;
|
|
}
|
|
|
|
/* Congestion Avoidance */
|
|
fqr.congestionAvoidanceEnable = congst_avoid_ena;
|
|
if (congst_avoid_ena) {
|
|
fqr.congestionAvoidanceParams.h_QmCg = congst_group;
|
|
fqr.congestionAvoidanceParams.overheadAccountingLength =
|
|
overhead_accounting_len;
|
|
fqr.congestionAvoidanceParams.fqTailDropThreshold =
|
|
tail_drop_threshold;
|
|
} else {
|
|
fqr.congestionAvoidanceParams.h_QmCg = 0;
|
|
fqr.congestionAvoidanceParams.overheadAccountingLength = 0;
|
|
fqr.congestionAvoidanceParams.fqTailDropThreshold = 0;
|
|
}
|
|
|
|
fqrh = QM_FQR_Create(&fqr);
|
|
if (fqrh == NULL) {
|
|
device_printf(sc->sc_dev, "could not create Frame Queue Range"
|
|
"\n");
|
|
goto err;
|
|
}
|
|
|
|
sc->sc_fqr_cpu[QM_FQR_GetFqid(fqrh)] = PCPU_GET(cpuid);
|
|
|
|
sched_unpin();
|
|
|
|
return (fqrh);
|
|
|
|
err:
|
|
sched_unpin();
|
|
|
|
return (NULL);
|
|
}
|
|
|
|
t_Error
|
|
qman_fqr_free(t_Handle fqr)
|
|
{
|
|
struct qman_softc *sc;
|
|
t_Error error;
|
|
|
|
sc = qman_sc;
|
|
thread_lock(curthread);
|
|
sched_bind(curthread, sc->sc_fqr_cpu[QM_FQR_GetFqid(fqr)]);
|
|
thread_unlock(curthread);
|
|
|
|
error = QM_FQR_Free(fqr);
|
|
|
|
thread_lock(curthread);
|
|
sched_unbind(curthread);
|
|
thread_unlock(curthread);
|
|
|
|
return (error);
|
|
}
|
|
|
|
t_Error
|
|
qman_fqr_register_cb(t_Handle fqr, t_QmReceivedFrameCallback *callback,
|
|
t_Handle app)
|
|
{
|
|
struct qman_softc *sc;
|
|
t_Error error;
|
|
t_Handle portal;
|
|
|
|
sc = qman_sc;
|
|
sched_pin();
|
|
|
|
/* Ensure we have got QMan port initialized */
|
|
portal = qman_portal_setup(sc);
|
|
if (portal == NULL) {
|
|
device_printf(sc->sc_dev, "could not setup QMan portal\n");
|
|
sched_unpin();
|
|
return (E_NOT_SUPPORTED);
|
|
}
|
|
|
|
error = QM_FQR_RegisterCB(fqr, callback, app);
|
|
|
|
sched_unpin();
|
|
|
|
return (error);
|
|
}
|
|
|
|
t_Error
|
|
qman_fqr_enqueue(t_Handle fqr, uint32_t fqid_off, t_DpaaFD *frame)
|
|
{
|
|
struct qman_softc *sc;
|
|
t_Error error;
|
|
t_Handle portal;
|
|
|
|
sc = qman_sc;
|
|
sched_pin();
|
|
|
|
/* Ensure we have got QMan port initialized */
|
|
portal = qman_portal_setup(sc);
|
|
if (portal == NULL) {
|
|
device_printf(sc->sc_dev, "could not setup QMan portal\n");
|
|
sched_unpin();
|
|
return (E_NOT_SUPPORTED);
|
|
}
|
|
|
|
error = QM_FQR_Enqueue(fqr, portal, fqid_off, frame);
|
|
|
|
sched_unpin();
|
|
|
|
return (error);
|
|
}
|
|
|
|
uint32_t
|
|
qman_fqr_get_counter(t_Handle fqr, uint32_t fqid_off,
|
|
e_QmFqrCounters counter)
|
|
{
|
|
struct qman_softc *sc;
|
|
uint32_t val;
|
|
t_Handle portal;
|
|
|
|
sc = qman_sc;
|
|
sched_pin();
|
|
|
|
/* Ensure we have got QMan port initialized */
|
|
portal = qman_portal_setup(sc);
|
|
if (portal == NULL) {
|
|
device_printf(sc->sc_dev, "could not setup QMan portal\n");
|
|
sched_unpin();
|
|
return (0);
|
|
}
|
|
|
|
val = QM_FQR_GetCounter(fqr, portal, fqid_off, counter);
|
|
|
|
sched_unpin();
|
|
|
|
return (val);
|
|
}
|
|
|
|
t_Error
|
|
qman_fqr_pull_frame(t_Handle fqr, uint32_t fqid_off, t_DpaaFD *frame)
|
|
{
|
|
struct qman_softc *sc;
|
|
t_Error error;
|
|
t_Handle portal;
|
|
|
|
sc = qman_sc;
|
|
sched_pin();
|
|
|
|
/* Ensure we have got QMan port initialized */
|
|
portal = qman_portal_setup(sc);
|
|
if (portal == NULL) {
|
|
device_printf(sc->sc_dev, "could not setup QMan portal\n");
|
|
sched_unpin();
|
|
return (E_NOT_SUPPORTED);
|
|
}
|
|
|
|
error = QM_FQR_PullFrame(fqr, portal, fqid_off, frame);
|
|
|
|
sched_unpin();
|
|
|
|
return (error);
|
|
}
|
|
|
|
uint32_t
|
|
qman_fqr_get_base_fqid(t_Handle fqr)
|
|
{
|
|
struct qman_softc *sc;
|
|
uint32_t val;
|
|
t_Handle portal;
|
|
|
|
sc = qman_sc;
|
|
sched_pin();
|
|
|
|
/* Ensure we have got QMan port initialized */
|
|
portal = qman_portal_setup(sc);
|
|
if (portal == NULL) {
|
|
device_printf(sc->sc_dev, "could not setup QMan portal\n");
|
|
sched_unpin();
|
|
return (0);
|
|
}
|
|
|
|
val = QM_FQR_GetFqid(fqr);
|
|
|
|
sched_unpin();
|
|
|
|
return (val);
|
|
}
|
|
|
|
t_Error
|
|
qman_poll(e_QmPortalPollSource source)
|
|
{
|
|
struct qman_softc *sc;
|
|
t_Error error;
|
|
t_Handle portal;
|
|
|
|
sc = qman_sc;
|
|
sched_pin();
|
|
|
|
/* Ensure we have got QMan port initialized */
|
|
portal = qman_portal_setup(sc);
|
|
if (portal == NULL) {
|
|
device_printf(sc->sc_dev, "could not setup QMan portal\n");
|
|
sched_unpin();
|
|
return (E_NOT_SUPPORTED);
|
|
}
|
|
|
|
error = QM_Poll(sc->sc_qh, source);
|
|
|
|
sched_unpin();
|
|
|
|
return (error);
|
|
}
|
|
|
|
/*
|
|
* TODO: add polling and/or congestion support.
|
|
*/
|
|
|
|
/** @} */
|