2  * Copyright (C) 2005 - 2008 ServerEngines
 
   5  * This program is free software; you can redistribute it and/or
 
   6  * modify it under the terms of the GNU General Public License version 2
 
   7  * as published by the Free Software Foundation.  The full GNU General
 
   8  * Public License is included in this distribution in the file called COPYING.
 
  10  * Contact Information:
 
  11  * linux-drivers@serverengines.com
 
  14  * 209 N. Fair Oaks Ave
 
  21  * Completion Queue Objects
 
  24  *============================================================================
 
  25  *                  P U B L I C  R O U T I N E S
 
  26  *============================================================================
 
  30     This routine creates a completion queue based on the client completion
 
  31     queue configuration information.
 
  34     FunctionObject      - Handle to a function object
 
  35     CqBaseVa            - Base VA for a the CQ ring
 
  36     NumEntries          - CEV_CQ_CNT_* values
 
  37     solEventEnable      - 0 = All CQEs can generate Events if CQ is eventable
 
  38                         1 = only CQEs with solicited bit set are eventable
 
  39     eventable           - Eventable CQ, generates interrupts.
 
  40     nodelay             - 1 = Force interrupt, relevent if CQ eventable.
 
  41                         Interrupt is asserted immediately after EQE
 
  42                         write is confirmed, regardless of EQ Timer
 
  43                         or watermark settings.
 
  44     wme                 - Enable watermark based coalescing
 
  45     wmThresh            - High watermark(CQ fullness at which event
 
  46                         or interrupt should be asserted).  These are the
 
  47                         CEV_WATERMARK encoded values.
 
  48     EqObject            - EQ Handle to assign to this CQ
 
  49     ppCqObject          - Internal CQ Handle returned.
 
  51     Returns BE_SUCCESS if successfull, otherwise a useful error code is
 
  57 int be_cq_create(struct be_function_object *pfob,
 
  58         struct ring_desc *rd, u32 length, bool solicited_eventable,
 
  59         bool no_delay, u32 wm_thresh,
 
  60         struct be_eq_object *eq_object, struct be_cq_object *cq_object)
 
  62         int status = BE_SUCCESS;
 
  63         u32 num_entries_encoding;
 
  64         u32 num_entries = length / sizeof(struct MCC_CQ_ENTRY_AMAP);
 
  65         struct FWCMD_COMMON_CQ_CREATE *fwcmd = NULL;
 
  66         struct MCC_WRB_AMAP *wrb = NULL;
 
  72         ASSERT(length % sizeof(struct MCC_CQ_ENTRY_AMAP) == 0);
 
  74         switch (num_entries) {
 
  76                 num_entries_encoding = CEV_CQ_CNT_256;
 
  79                 num_entries_encoding = CEV_CQ_CNT_512;
 
  82                 num_entries_encoding = CEV_CQ_CNT_1024;
 
  86                 return BE_STATUS_INVALID_PARAMETER;
 
  90          * All cq entries all the same size.  Use iSCSI version
 
  91          * as a test for the proper rd length.
 
  93         memset(cq_object, 0, sizeof(*cq_object));
 
  95         atomic_set(&cq_object->ref_count, 0);
 
  96         cq_object->parent_function = pfob;
 
  97         cq_object->eq_object = eq_object;
 
  98         cq_object->num_entries = num_entries;
 
  99         /* save for MCC cq processing */
 
 100         cq_object->va = rd->va;
 
 103         length = num_entries * sizeof(struct MCC_CQ_ENTRY_AMAP);
 
 105         spin_lock_irqsave(&pfob->post_lock, irql);
 
 107         wrb = be_function_peek_mcc_wrb(pfob);
 
 110                 TRACE(DL_ERR, "No free MCC WRBs in create EQ.");
 
 111                 status = BE_STATUS_NO_MCC_WRB;
 
 114         /* Prepares an embedded fwcmd, including request/response sizes. */
 
 115         fwcmd = BE_PREPARE_EMBEDDED_FWCMD(pfob, wrb, COMMON_CQ_CREATE);
 
 117         fwcmd->params.request.num_pages = PAGES_SPANNED(OFFSET_IN_PAGE(rd->va),
 
 120         AMAP_SET_BITS_PTR(CQ_CONTEXT, valid, &fwcmd->params.request.context, 1);
 
 121         n = pfob->pci_function_number;
 
 122         AMAP_SET_BITS_PTR(CQ_CONTEXT, Func, &fwcmd->params.request.context, n);
 
 124         n = (eq_object != NULL);
 
 125         AMAP_SET_BITS_PTR(CQ_CONTEXT, Eventable,
 
 126                                 &fwcmd->params.request.context, n);
 
 127         AMAP_SET_BITS_PTR(CQ_CONTEXT, Armed, &fwcmd->params.request.context, 1);
 
 129         n = eq_object ? eq_object->eq_id : 0;
 
 130         AMAP_SET_BITS_PTR(CQ_CONTEXT, EQID, &fwcmd->params.request.context, n);
 
 131         AMAP_SET_BITS_PTR(CQ_CONTEXT, Count,
 
 132                         &fwcmd->params.request.context, num_entries_encoding);
 
 134         n = 0; /* Protection Domain is always 0 in  Linux  driver */
 
 135         AMAP_SET_BITS_PTR(CQ_CONTEXT, PD, &fwcmd->params.request.context, n);
 
 136         AMAP_SET_BITS_PTR(CQ_CONTEXT, NoDelay,
 
 137                                 &fwcmd->params.request.context, no_delay);
 
 138         AMAP_SET_BITS_PTR(CQ_CONTEXT, SolEvent,
 
 139                         &fwcmd->params.request.context, solicited_eventable);
 
 141         n = (wm_thresh != 0xFFFFFFFF);
 
 142         AMAP_SET_BITS_PTR(CQ_CONTEXT, WME, &fwcmd->params.request.context, n);
 
 144         n = (n ? wm_thresh : 0);
 
 145         AMAP_SET_BITS_PTR(CQ_CONTEXT, Watermark,
 
 146                                 &fwcmd->params.request.context, n);
 
 147         /* Create a page list for the FWCMD. */
 
 148         be_rd_to_pa_list(rd, fwcmd->params.request.pages,
 
 149                           ARRAY_SIZE(fwcmd->params.request.pages));
 
 151         /* Post the f/w command */
 
 152         status = be_function_post_mcc_wrb(pfob, wrb, NULL, NULL, NULL,
 
 153                         NULL, NULL, fwcmd, NULL);
 
 154         if (status != BE_SUCCESS) {
 
 155                 TRACE(DL_ERR, "MCC to create CQ failed.");
 
 158         /* Remember the CQ id. */
 
 159         cq_object->cq_id = fwcmd->params.response.cq_id;
 
 161         /* insert this cq into eq_object reference */
 
 163                 atomic_inc(&eq_object->ref_count);
 
 164                 list_add_tail(&cq_object->cqlist_for_eq,
 
 165                                         &eq_object->cq_list_head);
 
 169         spin_unlock_irqrestore(&pfob->post_lock, irql);
 
 171         if (pfob->pend_queue_driving && pfob->mcc) {
 
 172                 pfob->pend_queue_driving = 0;
 
 173                 be_drive_mcc_wrb_queue(pfob->mcc);
 
 180     Deferences the given object. Once the object's reference count drops to
 
 181     zero, the object is destroyed and all resources that are held by this object
 
 182     are released.  The on-chip context is also destroyed along with the queue
 
 183     ID, and any mappings made into the UT.
 
 185     cq_object            - CQ handle returned from cq_object_create.
 
 187     returns the current reference count on the object
 
 189     IRQL: IRQL < DISPATCH_LEVEL
 
 191 int be_cq_destroy(struct be_cq_object *cq_object)
 
 195         /* Nothing should reference this CQ at this point. */
 
 196         ASSERT(atomic_read(&cq_object->ref_count) == 0);
 
 198         /* Send fwcmd to destroy the CQ. */
 
 199         status = be_function_ring_destroy(cq_object->parent_function,
 
 200                      cq_object->cq_id, FWCMD_RING_TYPE_CQ,
 
 201                                         NULL, NULL, NULL, NULL);
 
 204         /* Remove reference if this is an eventable CQ. */
 
 205         if (cq_object->eq_object) {
 
 206                 atomic_dec(&cq_object->eq_object->ref_count);
 
 207                 list_del(&cq_object->cqlist_for_eq);