The Design and Implementation of the FreeBSD Operating System, Second Edition
Now available: The Design and Implementation of the FreeBSD Operating System (Second Edition)


[ source navigation ] [ diff markup ] [ identifier search ] [ freetext search ] [ file search ] [ list types ] [ track identifier ]

FreeBSD/Linux Kernel Cross Reference
sys/dev/smartpqi/smartpqi_main.c

Version: -  FREEBSD  -  FREEBSD-13-STABLE  -  FREEBSD-13-0  -  FREEBSD-12-STABLE  -  FREEBSD-12-0  -  FREEBSD-11-STABLE  -  FREEBSD-11-0  -  FREEBSD-10-STABLE  -  FREEBSD-10-0  -  FREEBSD-9-STABLE  -  FREEBSD-9-0  -  FREEBSD-8-STABLE  -  FREEBSD-8-0  -  FREEBSD-7-STABLE  -  FREEBSD-7-0  -  FREEBSD-6-STABLE  -  FREEBSD-6-0  -  FREEBSD-5-STABLE  -  FREEBSD-5-0  -  FREEBSD-4-STABLE  -  FREEBSD-3-STABLE  -  FREEBSD22  -  l41  -  OPENBSD  -  linux-2.6  -  MK84  -  PLAN9  -  xnu-8792 
SearchContext: -  none  -  3  -  10 

    1 /*-
    2  * Copyright 2016-2021 Microchip Technology, Inc. and/or its subsidiaries.
    3  *
    4  * Redistribution and use in source and binary forms, with or without
    5  * modification, are permitted provided that the following conditions
    6  * are met:
    7  * 1. Redistributions of source code must retain the above copyright
    8  *    notice, this list of conditions and the following disclaimer.
    9  * 2. Redistributions in binary form must reproduce the above copyright
   10  *    notice, this list of conditions and the following disclaimer in the
   11  *    documentation and/or other materials provided with the distribution.
   12  *
   13  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
   14  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
   15  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
   16  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
   17  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
   18  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
   19  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
   20  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
   21  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
   22  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
   23  * SUCH DAMAGE.
   24  */
   25 
   26 /* $FreeBSD$ */
   27 
   28 /*
   29  * Driver for the Microsemi Smart storage controllers
   30  */
   31 
   32 #include "smartpqi_includes.h"
   33 #include "smartpqi_prototypes.h"
   34 
   35 CTASSERT(BSD_SUCCESS == PQI_STATUS_SUCCESS);
   36 
   37 /*
   38  * Supported devices
   39  */
   40 struct pqi_ident
   41 {
   42         u_int16_t               vendor;
   43         u_int16_t               device;
   44         u_int16_t               subvendor;
   45         u_int16_t               subdevice;
   46         int                     hwif;
   47         char                    *desc;
   48 } pqi_identifiers[] = {
   49         /* (MSCC PM8205 8x12G based) */
   50         {0x9005, 0x028f, 0x103c, 0x600,  PQI_HWIF_SRCV, "P408i-p SR Gen10"},
   51         {0x9005, 0x028f, 0x103c, 0x601,  PQI_HWIF_SRCV, "P408e-p SR Gen10"},
   52         {0x9005, 0x028f, 0x103c, 0x602,  PQI_HWIF_SRCV, "P408i-a SR Gen10"},
   53         {0x9005, 0x028f, 0x103c, 0x603,  PQI_HWIF_SRCV, "P408i-c SR Gen10"},
   54         {0x9005, 0x028f, 0x1028, 0x1FE0, PQI_HWIF_SRCV, "SmartRAID 3162-8i/eDell"},
   55         {0x9005, 0x028f, 0x9005, 0x608,  PQI_HWIF_SRCV, "SmartRAID 3162-8i/e"},
   56         {0x9005, 0x028f, 0x103c, 0x609,  PQI_HWIF_SRCV, "P408i-sb SR G10"},
   57 
   58         /* (MSCC PM8225 8x12G based) */
   59         {0x9005, 0x028f, 0x103c, 0x650,  PQI_HWIF_SRCV, "E208i-p SR Gen10"},
   60         {0x9005, 0x028f, 0x103c, 0x651,  PQI_HWIF_SRCV, "E208e-p SR Gen10"},
   61         {0x9005, 0x028f, 0x103c, 0x652,  PQI_HWIF_SRCV, "E208i-c SR Gen10"},
   62         {0x9005, 0x028f, 0x103c, 0x654,  PQI_HWIF_SRCV, "E208i-a SR Gen10"},
   63         {0x9005, 0x028f, 0x103c, 0x655,  PQI_HWIF_SRCV, "P408e-m SR Gen10"},
   64 
   65         /* (MSCC PM8221 8x12G based) */
   66         {0x9005, 0x028f, 0x103c, 0x700,  PQI_HWIF_SRCV, "P204i-c SR Gen10"},
   67         {0x9005, 0x028f, 0x103c, 0x701,  PQI_HWIF_SRCV, "P204i-b SR Gen10"},
   68         {0x9005, 0x028f, 0x193d, 0x1104, PQI_HWIF_SRCV, "UN RAID P2404-Mf-4i-2GB"},
   69         {0x9005, 0x028f, 0x193d, 0x1106, PQI_HWIF_SRCV, "UN RAID P2404-Mf-4i-1GB"},
   70         {0x9005, 0x028f, 0x193d, 0x1108, PQI_HWIF_SRCV, "UN RAID P4408-Ma-8i-2GB"},
   71 
   72 
   73         /* (MSCC PM8204 8x12G based) */
   74         {0x9005, 0x028f, 0x9005, 0x800,  PQI_HWIF_SRCV, "SmartRAID 3154-8i"},
   75         {0x9005, 0x028f, 0x9005, 0x801,  PQI_HWIF_SRCV, "SmartRAID 3152-8i"},
   76         {0x9005, 0x028f, 0x9005, 0x802,  PQI_HWIF_SRCV, "SmartRAID 3151-4i"},
   77         {0x9005, 0x028f, 0x9005, 0x803,  PQI_HWIF_SRCV, "SmartRAID 3101-4i"},
   78         {0x9005, 0x028f, 0x9005, 0x804,  PQI_HWIF_SRCV, "SmartRAID 3154-8e"},
   79         {0x9005, 0x028f, 0x9005, 0x805,  PQI_HWIF_SRCV, "SmartRAID 3102-8i"},
   80         {0x9005, 0x028f, 0x9005, 0x806,  PQI_HWIF_SRCV, "SmartRAID 3100"},
   81         {0x9005, 0x028f, 0x9005, 0x807,  PQI_HWIF_SRCV, "SmartRAID 3162-8i"},
   82         {0x9005, 0x028f, 0x152d, 0x8a22, PQI_HWIF_SRCV, "QS-8204-8i"},
   83         {0x9005, 0x028f, 0x193d, 0xf460, PQI_HWIF_SRCV, "UN RAID P460-M4"},
   84         {0x9005, 0x028f, 0x193d, 0xf461, PQI_HWIF_SRCV, "UN RAID P460-B4"},
   85         {0x9005, 0x028f, 0x1bd4, 0x004b, PQI_HWIF_SRCV, "INSPUR PM8204-2GB"},
   86         {0x9005, 0x028f, 0x1bd4, 0x004c, PQI_HWIF_SRCV, "INSPUR PM8204-4GB"},
   87         {0x9005, 0x028f, 0x193d, 0x1105, PQI_HWIF_SRCV, "UN RAID P4408-Mf-8i-2GB"},
   88         {0x9005, 0x028f, 0x193d, 0x1107, PQI_HWIF_SRCV, "UN RAID P4408-Mf-8i-4GB"},
   89         {0x9005, 0x028f, 0x1d8d, 0x800,  PQI_HWIF_SRCV, "Fiberhome SmartRAID AIS-8204-8i"},
   90         {0x9005, 0x028f, 0x9005, 0x0808, PQI_HWIF_SRCV, "SmartRAID 3101E-4i"},
   91         {0x9005, 0x028f, 0x9005, 0x0809, PQI_HWIF_SRCV, "SmartRAID 3102E-8i"},
   92         {0x9005, 0x028f, 0x9005, 0x080a, PQI_HWIF_SRCV, "SmartRAID 3152-8i/N"},
   93 
   94         /* (MSCC PM8222 8x12G based) */
   95         {0x9005, 0x028f, 0x9005, 0x900,  PQI_HWIF_SRCV, "SmartHBA 2100-8i"},
   96         {0x9005, 0x028f, 0x9005, 0x901,  PQI_HWIF_SRCV, "SmartHBA 2100-4i"},
   97         {0x9005, 0x028f, 0x9005, 0x902,  PQI_HWIF_SRCV, "HBA 1100-8i"},
   98         {0x9005, 0x028f, 0x9005, 0x903,  PQI_HWIF_SRCV, "HBA 1100-4i"},
   99         {0x9005, 0x028f, 0x9005, 0x904,  PQI_HWIF_SRCV, "SmartHBA 2100-8e"},
  100         {0x9005, 0x028f, 0x9005, 0x905,  PQI_HWIF_SRCV, "HBA 1100-8e"},
  101         {0x9005, 0x028f, 0x9005, 0x906,  PQI_HWIF_SRCV, "SmartHBA 2100-4i4e"},
  102         {0x9005, 0x028f, 0x9005, 0x907,  PQI_HWIF_SRCV, "HBA 1100"},
  103         {0x9005, 0x028f, 0x9005, 0x908,  PQI_HWIF_SRCV, "SmartHBA 2100"},
  104         {0x9005, 0x028f, 0x9005, 0x90a,  PQI_HWIF_SRCV, "SmartHBA 2100A-8i"},
  105         {0x9005, 0x028f, 0x193d, 0x8460, PQI_HWIF_SRCV, "UN HBA H460-M1"},
  106         {0x9005, 0x028f, 0x193d, 0x8461, PQI_HWIF_SRCV, "UN HBA H460-B1"},
  107         {0x9005, 0x028f, 0x193d, 0xc460, PQI_HWIF_SRCV, "UN RAID P460-M2"},
  108         {0x9005, 0x028f, 0x193d, 0xc461, PQI_HWIF_SRCV, "UN RAID P460-B2"},
  109         {0x9005, 0x028f, 0x1bd4, 0x004a, PQI_HWIF_SRCV, "INSPUR PM8222-SHBA"},
  110         {0x9005, 0x028f, 0x13fe, 0x8312, PQI_HWIF_SRCV, "MIC-8312BridgeB"},
  111         {0x9005, 0x028f, 0x1bd4, 0x004f, PQI_HWIF_SRCV, "INSPUR PM8222-HBA"},
  112         {0x9005, 0x028f, 0x1d8d, 0x908,  PQI_HWIF_SRCV, "Fiberhome SmartHBA AIS-8222-8i"},
  113         {0x9005, 0x028f, 0x1bd4, 0x006C, PQI_HWIF_SRCV, "INSPUR RS0800M5E8i"},
  114         {0x9005, 0x028f, 0x1bd4, 0x006D, PQI_HWIF_SRCV, "INSPUR RS0800M5H8i"},
  115 
  116         /* (SRCx MSCC FVB 24x12G based) */
  117         {0x9005, 0x028f, 0x103c, 0x1001, PQI_HWIF_SRCV, "MSCC FVB"},
  118 
  119         /* (MSCC PM8241 24x12G based) */
  120 
  121         /* (MSCC PM8242 24x12G based) */
  122         {0x9005, 0x028f, 0x152d, 0x8a37, PQI_HWIF_SRCV, "QS-8242-24i"},
  123         {0x9005, 0x028f, 0x9005, 0x1300, PQI_HWIF_SRCV, "HBA 1100-8i8e"},
  124         {0x9005, 0x028f, 0x9005, 0x1301, PQI_HWIF_SRCV, "HBA 1100-24i"},
  125         {0x9005, 0x028f, 0x9005, 0x1302, PQI_HWIF_SRCV, "SmartHBA 2100-8i8e"},
  126         {0x9005, 0x028f, 0x9005, 0x1303, PQI_HWIF_SRCV, "SmartHBA 2100-24i"},
  127         {0x9005, 0x028f, 0x105b, 0x1321, PQI_HWIF_SRCV, "8242-24i"},
  128         {0x9005, 0x028f, 0x1bd4, 0x0045, PQI_HWIF_SRCV, "INSPUR SMART-HBA 8242-24i"},
  129 
  130         /* (MSCC PM8236 16x12G based) */
  131         {0x9005, 0x028f, 0x152d, 0x8a24, PQI_HWIF_SRCV, "QS-8236-16i"},
  132         {0x9005, 0x028f, 0x9005, 0x1380, PQI_HWIF_SRCV, "SmartRAID 3154-16i"},
  133         {0x9005, 0x028f, 0x1bd4, 0x0046, PQI_HWIF_SRCV, "INSPUR RAID 8236-16i"},
  134         {0x9005, 0x028f, 0x1d8d, 0x806,  PQI_HWIF_SRCV, "Fiberhome SmartRAID AIS-8236-16i"},
  135         {0x9005, 0x028f, 0x1cf2, 0x5449, PQI_HWIF_SRCV, "ZTE SmartROC3100 RS241-18i 2G"},
  136         {0x9005, 0x028f, 0x1cf2, 0x544A, PQI_HWIF_SRCV, "ZTE SmartROC3100 RS242-18i 4G"},
  137         {0x9005, 0x028f, 0x1cf2, 0x544D, PQI_HWIF_SRCV, "ZTE SmartROC3100 RM241B-18i 2G"},
  138         {0x9005, 0x028f, 0x1cf2, 0x544E, PQI_HWIF_SRCV, "ZTE SmartROC3100 RM242B-18i 4G"},
  139 
  140         /* (MSCC PM8237 24x12G based) */
  141         {0x9005, 0x028f, 0x103c, 0x1100, PQI_HWIF_SRCV, "P816i-a SR Gen10"},
  142         {0x9005, 0x028f, 0x103c, 0x1101, PQI_HWIF_SRCV, "P416ie-m SR G10"},
  143 
  144         /* (MSCC PM8238 16x12G based) */
  145         {0x9005, 0x028f, 0x152d, 0x8a23, PQI_HWIF_SRCV, "QS-8238-16i"},
  146         {0x9005, 0x028f, 0x9005, 0x1280, PQI_HWIF_SRCV, "HBA 1100-16i"},
  147         {0x9005, 0x028f, 0x9005, 0x1281, PQI_HWIF_SRCV, "HBA 1100-16e"},
  148         {0x9005, 0x028f, 0x105b, 0x1211, PQI_HWIF_SRCV, "8238-16i"},
  149         {0x9005, 0x028f, 0x1bd4, 0x0048, PQI_HWIF_SRCV, "INSPUR SMART-HBA 8238-16i"},
  150         {0x9005, 0x028f, 0x9005, 0x1282, PQI_HWIF_SRCV, "SmartHBA 2100-16i"},
  151         {0x9005, 0x028f, 0x1d8d, 0x916,  PQI_HWIF_SRCV, "Fiberhome SmartHBA AIS-8238-16i"},
  152         {0x9005, 0x028f, 0x1458, 0x1000, PQI_HWIF_SRCV, "GIGABYTE SmartHBA CLN1832"},
  153         {0x9005, 0x028f, 0x1cf2, 0x544F, PQI_HWIF_SRCV, "ZTE SmartIOC2100 RM243B-18i"},
  154 
  155         /* (MSCC PM8240 24x12G based) */
  156         {0x9005, 0x028f, 0x152d, 0x8a36, PQI_HWIF_SRCV, "QS-8240-24i"},
  157         {0x9005, 0x028f, 0x9005, 0x1200, PQI_HWIF_SRCV, "SmartRAID 3154-24i"},
  158         {0x9005, 0x028f, 0x9005, 0x1201, PQI_HWIF_SRCV, "SmartRAID 3154-8i16e"},
  159         {0x9005, 0x028f, 0x9005, 0x1202, PQI_HWIF_SRCV, "SmartRAID 3154-8i8e"},
  160         {0x9005, 0x028f, 0x1bd4, 0x0047, PQI_HWIF_SRCV, "INSPUR RAID 8240-24i"},
  161         {0x9005, 0x028f, 0x1F0C, 0x3161, PQI_HWIF_SRCV, "NT RAID 3100-24i"},
  162 
  163         /* Huawei ID's */
  164         {0x9005, 0x028f, 0x19e5, 0xd227, PQI_HWIF_SRCV, "SR465C-M 4G"},
  165         {0x9005, 0x028f, 0x19e5, 0xd22a, PQI_HWIF_SRCV, "SR765-M"},
  166         {0x9005, 0x028f, 0x19e5, 0xd228, PQI_HWIF_SRCV, "SR455C-M 2G"},
  167         {0x9005, 0x028f, 0x19e5, 0xd22c, PQI_HWIF_SRCV, "SR455C-M 4G"},
  168         {0x9005, 0x028f, 0x19e5, 0xd229, PQI_HWIF_SRCV, "SR155-M"},
  169         {0x9005, 0x028f, 0x19e5, 0xd22b, PQI_HWIF_SRCV, "SR455C-ME 4G"},
  170         /* (MSCC PM8254 32x12G based) */
  171         {0x9005, 0x028f, 0x9005, 0x14a2, PQI_HWIF_SRCV, "SmartRAID 3252-8i"},
  172         {0x9005, 0x028f, 0x9005, 0x14a4, PQI_HWIF_SRCV, "SmartRAID 3254-8i /e"},
  173         {0x9005, 0x028f, 0x9005, 0x14a5, PQI_HWIF_SRCV, "SmartRAID 3252-8i /e"},
  174         {0x9005, 0x028f, 0x9005, 0x14a6, PQI_HWIF_SRCV, "SmartRAID 3204-8i /e"},
  175 /* (MSCC PM8265 16x12G based) */
  176         {0x9005, 0x028f, 0x9005, 0x1474, PQI_HWIF_SRCV, "SmartRAID 3254-16io /e"},
  177 /* (MSCC PM8270 16x12G based) */
  178         {0x9005, 0x028f, 0x9005, 0x1463, PQI_HWIF_SRCV, "SmartHBA 2200-8io /e"},
  179         {0x9005, 0x028f, 0x9005, 0x14c2, PQI_HWIF_SRCV, "SmartHBA 2200-16io /e"},
  180         /* (MSCC PM8279 32x12G based) */
  181         {0x9005, 0x028f, 0x1590, 0x0381, PQI_HWIF_SRCV, "SR932i-p Gen11"},
  182         {0x9005, 0x028f, 0x1590, 0x0382, PQI_HWIF_SRCV, "SR308i-p Gen11"},
  183         {0x9005, 0x028f, 0x1590, 0x0383, PQI_HWIF_SRCV, "SR308i-o Gen11"},
  184         {0x9005, 0x028f, 0x1590, 0x02db, PQI_HWIF_SRCV, "SR416ie-m Gen11"},
  185         {0x9005, 0x028f, 0x1590, 0x032e, PQI_HWIF_SRCV, "SR416i-o Gen11"},
  186 
  187         {0, 0, 0, 0, 0, 0}
  188 };
  189 
  190 struct pqi_ident
  191 pqi_family_identifiers[] = {
  192         {0x9005, 0x028f, 0, 0, PQI_HWIF_SRCV, "Smart Array Storage Controller"},
  193         {0, 0, 0, 0, 0, 0}
  194 };
  195 
  196 /*
  197  * Function to identify the installed adapter.
  198  */
  199 static struct
  200 pqi_ident *pqi_find_ident(device_t dev)
  201 {
  202         struct pqi_ident *m;
  203         u_int16_t vendid, devid, sub_vendid, sub_devid;
  204 
  205         vendid = pci_get_vendor(dev);
  206         devid = pci_get_device(dev);
  207         sub_vendid = pci_get_subvendor(dev);
  208         sub_devid = pci_get_subdevice(dev);
  209 
  210         for (m = pqi_identifiers; m->vendor != 0; m++) {
  211                 if ((m->vendor == vendid) && (m->device == devid) &&
  212                         (m->subvendor == sub_vendid) &&
  213                         (m->subdevice == sub_devid)) {
  214                         return (m);
  215                 }
  216         }
  217 
  218         for (m = pqi_family_identifiers; m->vendor != 0; m++) {
  219                 if ((m->vendor == vendid) && (m->device == devid)) {
  220                         return (m);
  221                 }
  222         }
  223 
  224         return (NULL);
  225 }
  226 
  227 /*
  228  * Determine whether this is one of our supported adapters.
  229  */
  230 static int
  231 smartpqi_probe(device_t dev)
  232 {
  233         struct pqi_ident *id;
  234 
  235         if ((id = pqi_find_ident(dev)) != NULL) {
  236                 device_set_desc(dev, id->desc);
  237                 return(BUS_PROBE_VENDOR);
  238         }
  239 
  240         return(ENXIO);
  241 }
  242 
  243 /*
  244  * Store Bus/Device/Function in softs
  245  */
  246 void
  247 pqisrc_save_controller_info(struct pqisrc_softstate *softs)
  248 {
  249         device_t dev = softs->os_specific.pqi_dev;
  250 
  251         softs->bus_id = (uint32_t)pci_get_bus(dev);
  252         softs->device_id = (uint32_t)pci_get_device(dev);
  253         softs->func_id = (uint32_t)pci_get_function(dev);
  254 }
  255 
  256 
  257 /*
  258  * Allocate resources for our device, set up the bus interface.
  259  * Initialize the PQI related functionality, scan devices, register sim to
  260  * upper layer, create management interface device node etc.
  261  */
  262 static int
  263 smartpqi_attach(device_t dev)
  264 {
  265         struct pqisrc_softstate *softs = NULL;
  266         struct pqi_ident *id = NULL;
  267         int error = BSD_SUCCESS;
  268         u_int32_t command = 0, i = 0;
  269         int card_index = device_get_unit(dev);
  270         rcb_t *rcbp = NULL;
  271 
  272         /*
  273          * Initialise softc.
  274          */
  275         softs = device_get_softc(dev);
  276 
  277         if (!softs) {
  278                 printf("Could not get softc\n");
  279                 error = EINVAL;
  280                 goto out;
  281         }
  282         memset(softs, 0, sizeof(*softs));
  283         softs->os_specific.pqi_dev = dev;
  284 
  285         DBG_FUNC("IN\n");
  286 
  287         /* assume failure is 'not configured' */
  288         error = ENXIO;
  289 
  290         /*
  291          * Verify that the adapter is correctly set up in PCI space.
  292          */
  293         pci_enable_busmaster(softs->os_specific.pqi_dev);
  294         command = pci_read_config(softs->os_specific.pqi_dev, PCIR_COMMAND, 2);
  295         if ((command & PCIM_CMD_MEMEN) == 0) {
  296                 DBG_ERR("memory window not available command = %d\n", command);
  297                 error = ENXIO;
  298                 goto out;
  299         }
  300 
  301         /*
  302          * Detect the hardware interface version, set up the bus interface
  303          * indirection.
  304          */
  305         id = pqi_find_ident(dev);
  306         if (!id) {
  307                 DBG_ERR("NULL return value from pqi_find_ident\n");
  308                 goto out;
  309         }
  310 
  311         softs->os_specific.pqi_hwif = id->hwif;
  312 
  313         switch(softs->os_specific.pqi_hwif) {
  314                 case PQI_HWIF_SRCV:
  315                         DBG_INFO("set hardware up for PMC SRCv for %p\n", softs);
  316                         break;
  317                 default:
  318                         softs->os_specific.pqi_hwif = PQI_HWIF_UNKNOWN;
  319                         DBG_ERR("unknown hardware type\n");
  320                         error = ENXIO;
  321                         goto out;
  322         }
  323 
  324         pqisrc_save_controller_info(softs);
  325 
  326         /*
  327          * Allocate the PCI register window.
  328          */
  329         softs->os_specific.pqi_regs_rid0 = PCIR_BAR(0);
  330         if ((softs->os_specific.pqi_regs_res0 =
  331                 bus_alloc_resource_any(softs->os_specific.pqi_dev, SYS_RES_MEMORY,
  332                 &softs->os_specific.pqi_regs_rid0, RF_ACTIVE)) == NULL) {
  333                 DBG_ERR("couldn't allocate register window 0\n");
  334                 /* assume failure is 'out of memory' */
  335                 error = ENOMEM;
  336                 goto out;
  337         }
  338 
  339         bus_get_resource_start(softs->os_specific.pqi_dev, SYS_RES_MEMORY,
  340                 softs->os_specific.pqi_regs_rid0);
  341 
  342         softs->pci_mem_handle.pqi_btag = rman_get_bustag(softs->os_specific.pqi_regs_res0);
  343         softs->pci_mem_handle.pqi_bhandle = rman_get_bushandle(softs->os_specific.pqi_regs_res0);
  344         /* softs->pci_mem_base_vaddr = (uintptr_t)rman_get_virtual(softs->os_specific.pqi_regs_res0); */
  345         softs->pci_mem_base_vaddr = (char *)rman_get_virtual(softs->os_specific.pqi_regs_res0);
  346 
  347         /*
  348          * Allocate the parent bus DMA tag appropriate for our PCI interface.
  349          *
  350          * Note that some of these controllers are 64-bit capable.
  351          */
  352         if (bus_dma_tag_create(bus_get_dma_tag(dev),    /* parent */
  353                                 PAGE_SIZE, 0,           /* algnmnt, boundary */
  354                                 BUS_SPACE_MAXADDR,/* lowaddr */
  355                                 BUS_SPACE_MAXADDR,      /* highaddr */
  356                                 NULL, NULL,             /* filter, filterarg */
  357                                 BUS_SPACE_MAXSIZE,      /* maxsize */
  358                                 BUS_SPACE_UNRESTRICTED, /* nsegments */
  359                                 BUS_SPACE_MAXSIZE,      /* maxsegsize */
  360                                 0,                      /* flags */
  361                                 NULL, NULL,             /* No locking needed */
  362                                 &softs->os_specific.pqi_parent_dmat)) {
  363                 DBG_ERR("can't allocate parent DMA tag\n");
  364                 /* assume failure is 'out of memory' */
  365                 error = ENOMEM;
  366                 goto dma_out;
  367         }
  368 
  369         softs->os_specific.sim_registered = FALSE;
  370         softs->os_name = "FreeBSD ";
  371 
  372         /* Initialize the PQI library */
  373         error = pqisrc_init(softs);
  374         if (error != PQI_STATUS_SUCCESS) {
  375                 DBG_ERR("Failed to initialize pqi lib error = %d\n", error);
  376                 error = ENXIO;
  377                 goto out;
  378         }
  379         else {
  380                 error = BSD_SUCCESS;
  381         }
  382 
  383     mtx_init(&softs->os_specific.cam_lock, "cam_lock", NULL, MTX_DEF);
  384     softs->os_specific.mtx_init = TRUE;
  385     mtx_init(&softs->os_specific.map_lock, "map_lock", NULL, MTX_DEF);
  386 
  387     callout_init(&softs->os_specific.wellness_periodic, 1);
  388     callout_init(&softs->os_specific.heartbeat_timeout_id, 1);
  389 
  390     /*
  391      * Create DMA tag for mapping buffers into controller-addressable space.
  392      */
  393     if (bus_dma_tag_create(softs->os_specific.pqi_parent_dmat,/* parent */
  394                                 PAGE_SIZE, 0,           /* algnmnt, boundary */
  395                                 BUS_SPACE_MAXADDR,/* lowaddr */
  396                                 BUS_SPACE_MAXADDR,      /* highaddr */
  397                                 NULL, NULL,             /* filter, filterarg */
  398                                 (bus_size_t)softs->pqi_cap.max_sg_elem*PAGE_SIZE,/* maxsize */
  399                                 softs->pqi_cap.max_sg_elem,     /* nsegments */
  400                                 BUS_SPACE_MAXSIZE,      /* maxsegsize */
  401                                 BUS_DMA_ALLOCNOW,               /* flags */
  402                                 busdma_lock_mutex,              /* lockfunc */
  403                                 &softs->os_specific.map_lock,   /* lockfuncarg*/
  404                                 &softs->os_specific.pqi_buffer_dmat)) {
  405                 DBG_ERR("can't allocate buffer DMA tag for pqi_buffer_dmat\n");
  406                 return (ENOMEM);
  407         }
  408 
  409         rcbp = &softs->rcb[1];
  410         for( i = 1;  i <= softs->pqi_cap.max_outstanding_io; i++, rcbp++ ) {
  411                 if ((error = bus_dmamap_create(softs->os_specific.pqi_buffer_dmat, 0, &rcbp->cm_datamap)) != 0) {
  412                         DBG_ERR("Cant create datamap for buf @"
  413                         "rcbp = %p maxio = %d error = %d\n",
  414                         rcbp, softs->pqi_cap.max_outstanding_io, error);
  415                         goto dma_out;
  416                 }
  417         }
  418 
  419         os_start_heartbeat_timer((void *)softs); /* Start the heart-beat timer */
  420         callout_reset(&softs->os_specific.wellness_periodic, 120 * hz,
  421                         os_wellness_periodic, softs);
  422 
  423         error = pqisrc_scan_devices(softs);
  424         if (error != PQI_STATUS_SUCCESS) {
  425                 DBG_ERR("Failed to scan lib error = %d\n", error);
  426                 error = ENXIO;
  427                 goto out;
  428         }
  429 
  430         error = register_sim(softs, card_index);
  431         if (error) {
  432                 DBG_ERR("Failed to register sim index = %d error = %d\n",
  433                         card_index, error);
  434                 goto out;
  435         }
  436 
  437         smartpqi_target_rescan(softs);
  438 
  439         TASK_INIT(&softs->os_specific.event_task, 0, pqisrc_event_worker,softs);
  440 
  441         error = create_char_dev(softs, card_index);
  442         if (error) {
  443                 DBG_ERR("Failed to register character device index=%d r=%d\n",
  444                         card_index, error);
  445                 goto out;
  446         }
  447         goto out;
  448 
  449 dma_out:
  450         if (softs->os_specific.pqi_regs_res0 != NULL)
  451                 bus_release_resource(softs->os_specific.pqi_dev, SYS_RES_MEMORY,
  452                         softs->os_specific.pqi_regs_rid0,
  453                         softs->os_specific.pqi_regs_res0);
  454 out:
  455         DBG_FUNC("OUT error = %d\n", error);
  456         return(error);
  457 }
  458 
  459 /*
  460  * Deallocate resources for our device.
  461  */
  462 static int
  463 smartpqi_detach(device_t dev)
  464 {
  465         struct pqisrc_softstate *softs = device_get_softc(dev);
  466         int rval = BSD_SUCCESS;
  467 
  468         DBG_FUNC("IN\n");
  469 
  470         if (softs == NULL)
  471                 return ENXIO;
  472 
  473         /* kill the periodic event */
  474         callout_drain(&softs->os_specific.wellness_periodic);
  475         /* Kill the heart beat event */
  476         callout_drain(&softs->os_specific.heartbeat_timeout_id);
  477 
  478         if (!pqisrc_ctrl_offline(softs)) {
  479                 rval = pqisrc_flush_cache(softs, PQISRC_NONE_CACHE_FLUSH_ONLY);
  480                 if (rval != PQI_STATUS_SUCCESS) {
  481                         DBG_ERR("Unable to flush adapter cache! rval = %d\n", rval);
  482                         rval = EIO;
  483                 }
  484         }
  485 
  486         destroy_char_dev(softs);
  487         pqisrc_uninit(softs);
  488         deregister_sim(softs);
  489         pci_release_msi(dev);
  490 
  491         DBG_FUNC("OUT\n");
  492 
  493         return rval;
  494 }
  495 
  496 /*
  497  * Bring the controller to a quiescent state, ready for system suspend.
  498  */
  499 static int
  500 smartpqi_suspend(device_t dev)
  501 {
  502         struct pqisrc_softstate *softs = device_get_softc(dev);
  503 
  504         DBG_FUNC("IN\n");
  505 
  506         if (softs == NULL)
  507                 return ENXIO;
  508 
  509         DBG_INFO("Suspending the device %p\n", softs);
  510         softs->os_specific.pqi_state |= SMART_STATE_SUSPEND;
  511 
  512         DBG_FUNC("OUT\n");
  513 
  514         return BSD_SUCCESS;
  515 }
  516 
  517 /*
  518  * Bring the controller back to a state ready for operation.
  519  */
  520 static int
  521 smartpqi_resume(device_t dev)
  522 {
  523         struct pqisrc_softstate *softs = device_get_softc(dev);
  524 
  525         DBG_FUNC("IN\n");
  526 
  527         if (softs == NULL)
  528                 return ENXIO;
  529 
  530         softs->os_specific.pqi_state &= ~SMART_STATE_SUSPEND;
  531 
  532         DBG_FUNC("OUT\n");
  533 
  534         return BSD_SUCCESS;
  535 }
  536 
  537 /*
  538  * Do whatever is needed during a system shutdown.
  539  */
  540 static int
  541 smartpqi_shutdown(device_t dev)
  542 {
  543         struct pqisrc_softstate *softs = device_get_softc(dev);
  544         int bsd_status = BSD_SUCCESS;
  545         int pqi_status;
  546 
  547         DBG_FUNC("IN\n");
  548 
  549         if (softs == NULL)
  550                 return ENXIO;
  551 
  552         if (pqisrc_ctrl_offline(softs))
  553                 return BSD_SUCCESS;
  554 
  555         pqi_status = pqisrc_flush_cache(softs, PQISRC_SHUTDOWN);
  556         if (pqi_status != PQI_STATUS_SUCCESS) {
  557                 DBG_ERR("Unable to flush adapter cache! rval = %d\n", pqi_status);
  558                 bsd_status = EIO;
  559         }
  560 
  561         DBG_FUNC("OUT\n");
  562                 
  563         return bsd_status;
  564 }
  565 
  566 /*
  567  * PCI bus interface.
  568  */
  569 static device_method_t pqi_methods[] = {
  570         /* Device interface */
  571         DEVMETHOD(device_probe,         smartpqi_probe),
  572         DEVMETHOD(device_attach,        smartpqi_attach),
  573         DEVMETHOD(device_detach,        smartpqi_detach),
  574         DEVMETHOD(device_suspend,       smartpqi_suspend),
  575         DEVMETHOD(device_resume,        smartpqi_resume),
  576         DEVMETHOD(device_shutdown,      smartpqi_shutdown),
  577         { 0, 0 }
  578 };
  579 
  580 static driver_t smartpqi_pci_driver = {
  581         "smartpqi",
  582         pqi_methods,
  583         sizeof(struct pqisrc_softstate)
  584 };
  585 
  586 DRIVER_MODULE(smartpqi, pci, smartpqi_pci_driver, 0, 0);
  587 MODULE_DEPEND(smartpqi, pci, 1, 1, 1);

Cache object: 2875bf1a608956e5e7c488f4eb19221b


[ source navigation ] [ diff markup ] [ identifier search ] [ freetext search ] [ file search ] [ list types ] [ track identifier ]


This page is part of the FreeBSD/Linux Linux Kernel Cross-Reference, and was automatically generated using a modified version of the LXR engine.