1f10a77bbSDavid C Somayajulu /* 2*35291c22SDavid C Somayajulu * Copyright (c) 2013-2016 Qlogic Corporation 3f10a77bbSDavid C Somayajulu * All rights reserved. 4f10a77bbSDavid C Somayajulu * 5f10a77bbSDavid C Somayajulu * Redistribution and use in source and binary forms, with or without 6f10a77bbSDavid C Somayajulu * modification, are permitted provided that the following conditions 7f10a77bbSDavid C Somayajulu * are met: 8f10a77bbSDavid C Somayajulu * 9f10a77bbSDavid C Somayajulu * 1. Redistributions of source code must retain the above copyright 10f10a77bbSDavid C Somayajulu * notice, this list of conditions and the following disclaimer. 11f10a77bbSDavid C Somayajulu * 2. Redistributions in binary form must reproduce the above copyright 12f10a77bbSDavid C Somayajulu * notice, this list of conditions and the following disclaimer in the 13f10a77bbSDavid C Somayajulu * documentation and/or other materials provided with the distribution. 14f10a77bbSDavid C Somayajulu * 15f10a77bbSDavid C Somayajulu * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" 16f10a77bbSDavid C Somayajulu * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 17f10a77bbSDavid C Somayajulu * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 18f10a77bbSDavid C Somayajulu * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE 19f10a77bbSDavid C Somayajulu * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR 20f10a77bbSDavid C Somayajulu * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF 21f10a77bbSDavid C Somayajulu * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 22f10a77bbSDavid C Somayajulu * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 23f10a77bbSDavid C Somayajulu * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 24f10a77bbSDavid C Somayajulu * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 25f10a77bbSDavid C Somayajulu * POSSIBILITY OF SUCH DAMAGE. 26f10a77bbSDavid C Somayajulu */ 27f10a77bbSDavid C Somayajulu /* 28f10a77bbSDavid C Somayajulu * File: ql_ioctl.c 29f10a77bbSDavid C Somayajulu * Author : David C Somayajulu, Qlogic Corporation, Aliso Viejo, CA 92656. 30f10a77bbSDavid C Somayajulu */ 31f10a77bbSDavid C Somayajulu 32f10a77bbSDavid C Somayajulu #include <sys/cdefs.h> 33f10a77bbSDavid C Somayajulu __FBSDID("$FreeBSD$"); 34f10a77bbSDavid C Somayajulu 35f10a77bbSDavid C Somayajulu 36f10a77bbSDavid C Somayajulu #include "ql_os.h" 37f10a77bbSDavid C Somayajulu #include "ql_hw.h" 38f10a77bbSDavid C Somayajulu #include "ql_def.h" 39f10a77bbSDavid C Somayajulu #include "ql_inline.h" 40f10a77bbSDavid C Somayajulu #include "ql_glbl.h" 41f10a77bbSDavid C Somayajulu #include "ql_ioctl.h" 42f10a77bbSDavid C Somayajulu 43f10a77bbSDavid C Somayajulu static int ql_eioctl(struct cdev *dev, u_long cmd, caddr_t data, int fflag, 44f10a77bbSDavid C Somayajulu struct thread *td); 45f10a77bbSDavid C Somayajulu 46f10a77bbSDavid C Somayajulu static struct cdevsw qla_cdevsw = { 47f10a77bbSDavid C Somayajulu .d_version = D_VERSION, 48f10a77bbSDavid C Somayajulu .d_ioctl = ql_eioctl, 49f10a77bbSDavid C Somayajulu .d_name = "qlcnic", 50f10a77bbSDavid C Somayajulu }; 51f10a77bbSDavid C Somayajulu 52f10a77bbSDavid C Somayajulu int 53f10a77bbSDavid C Somayajulu ql_make_cdev(qla_host_t *ha) 54f10a77bbSDavid C Somayajulu { 55f10a77bbSDavid C Somayajulu ha->ioctl_dev = make_dev(&qla_cdevsw, 56f10a77bbSDavid C Somayajulu ha->ifp->if_dunit, 57f10a77bbSDavid C Somayajulu UID_ROOT, 58f10a77bbSDavid C Somayajulu GID_WHEEL, 59f10a77bbSDavid C Somayajulu 0600, 60f10a77bbSDavid C Somayajulu "%s", 61f10a77bbSDavid C Somayajulu if_name(ha->ifp)); 62f10a77bbSDavid C Somayajulu 63f10a77bbSDavid C Somayajulu if (ha->ioctl_dev == NULL) 64f10a77bbSDavid C Somayajulu return (-1); 65f10a77bbSDavid C Somayajulu 66f10a77bbSDavid C Somayajulu ha->ioctl_dev->si_drv1 = ha; 67f10a77bbSDavid C Somayajulu 68f10a77bbSDavid C Somayajulu return (0); 69f10a77bbSDavid C Somayajulu } 70f10a77bbSDavid C Somayajulu 71f10a77bbSDavid C Somayajulu void 72f10a77bbSDavid C Somayajulu ql_del_cdev(qla_host_t *ha) 73f10a77bbSDavid C Somayajulu { 74f10a77bbSDavid C Somayajulu if (ha->ioctl_dev != NULL) 75f10a77bbSDavid C Somayajulu destroy_dev(ha->ioctl_dev); 76f10a77bbSDavid C Somayajulu return; 77f10a77bbSDavid C Somayajulu } 78f10a77bbSDavid C Somayajulu 79f10a77bbSDavid C Somayajulu static int 80f10a77bbSDavid C Somayajulu ql_eioctl(struct cdev *dev, u_long cmd, caddr_t data, int fflag, 81f10a77bbSDavid C Somayajulu struct thread *td) 82f10a77bbSDavid C Somayajulu { 83f10a77bbSDavid C Somayajulu qla_host_t *ha; 84f10a77bbSDavid C Somayajulu int rval = 0; 85f10a77bbSDavid C Somayajulu device_t pci_dev; 86f10a77bbSDavid C Somayajulu struct ifnet *ifp; 87f10a77bbSDavid C Somayajulu 88f10a77bbSDavid C Somayajulu q80_offchip_mem_val_t val; 89f10a77bbSDavid C Somayajulu qla_rd_pci_ids_t *pci_ids; 90f10a77bbSDavid C Somayajulu qla_rd_fw_dump_t *fw_dump; 91f10a77bbSDavid C Somayajulu union { 92f10a77bbSDavid C Somayajulu qla_reg_val_t *rv; 93f10a77bbSDavid C Somayajulu qla_rd_flash_t *rdf; 94f10a77bbSDavid C Somayajulu qla_wr_flash_t *wrf; 95f10a77bbSDavid C Somayajulu qla_erase_flash_t *erf; 96f10a77bbSDavid C Somayajulu qla_offchip_mem_val_t *mem; 97f10a77bbSDavid C Somayajulu } u; 98f10a77bbSDavid C Somayajulu 99f10a77bbSDavid C Somayajulu 100f10a77bbSDavid C Somayajulu if ((ha = (qla_host_t *)dev->si_drv1) == NULL) 101f10a77bbSDavid C Somayajulu return ENXIO; 102f10a77bbSDavid C Somayajulu 103f10a77bbSDavid C Somayajulu pci_dev= ha->pci_dev; 104f10a77bbSDavid C Somayajulu 105f10a77bbSDavid C Somayajulu switch(cmd) { 106f10a77bbSDavid C Somayajulu 107f10a77bbSDavid C Somayajulu case QLA_RDWR_REG: 108f10a77bbSDavid C Somayajulu 109f10a77bbSDavid C Somayajulu u.rv = (qla_reg_val_t *)data; 110f10a77bbSDavid C Somayajulu 111f10a77bbSDavid C Somayajulu if (u.rv->direct) { 112f10a77bbSDavid C Somayajulu if (u.rv->rd) { 113f10a77bbSDavid C Somayajulu u.rv->val = READ_REG32(ha, u.rv->reg); 114f10a77bbSDavid C Somayajulu } else { 115f10a77bbSDavid C Somayajulu WRITE_REG32(ha, u.rv->reg, u.rv->val); 116f10a77bbSDavid C Somayajulu } 117f10a77bbSDavid C Somayajulu } else { 118f10a77bbSDavid C Somayajulu if ((rval = ql_rdwr_indreg32(ha, u.rv->reg, &u.rv->val, 119f10a77bbSDavid C Somayajulu u.rv->rd))) 120f10a77bbSDavid C Somayajulu rval = ENXIO; 121f10a77bbSDavid C Somayajulu } 122f10a77bbSDavid C Somayajulu break; 123f10a77bbSDavid C Somayajulu 124f10a77bbSDavid C Somayajulu case QLA_RD_FLASH: 125f10a77bbSDavid C Somayajulu 126f10a77bbSDavid C Somayajulu if (!ha->hw.flags.fdt_valid) { 127f10a77bbSDavid C Somayajulu rval = EIO; 128f10a77bbSDavid C Somayajulu break; 129f10a77bbSDavid C Somayajulu } 130f10a77bbSDavid C Somayajulu 131f10a77bbSDavid C Somayajulu u.rdf = (qla_rd_flash_t *)data; 132f10a77bbSDavid C Somayajulu if ((rval = ql_rd_flash32(ha, u.rdf->off, &u.rdf->data))) 133f10a77bbSDavid C Somayajulu rval = ENXIO; 134f10a77bbSDavid C Somayajulu break; 135f10a77bbSDavid C Somayajulu 136f10a77bbSDavid C Somayajulu case QLA_WR_FLASH: 137f10a77bbSDavid C Somayajulu 138f10a77bbSDavid C Somayajulu ifp = ha->ifp; 139f10a77bbSDavid C Somayajulu 140f10a77bbSDavid C Somayajulu if (ifp == NULL) { 141f10a77bbSDavid C Somayajulu rval = ENXIO; 142f10a77bbSDavid C Somayajulu break; 143f10a77bbSDavid C Somayajulu } 144f10a77bbSDavid C Somayajulu 145f10a77bbSDavid C Somayajulu if (ifp->if_drv_flags & (IFF_DRV_OACTIVE | IFF_DRV_RUNNING)) { 146f10a77bbSDavid C Somayajulu rval = ENXIO; 147f10a77bbSDavid C Somayajulu break; 148f10a77bbSDavid C Somayajulu } 149f10a77bbSDavid C Somayajulu 150f10a77bbSDavid C Somayajulu if (!ha->hw.flags.fdt_valid) { 151f10a77bbSDavid C Somayajulu rval = EIO; 152f10a77bbSDavid C Somayajulu break; 153f10a77bbSDavid C Somayajulu } 154f10a77bbSDavid C Somayajulu 155f10a77bbSDavid C Somayajulu u.wrf = (qla_wr_flash_t *)data; 156f10a77bbSDavid C Somayajulu if ((rval = ql_wr_flash_buffer(ha, u.wrf->off, u.wrf->size, 157f10a77bbSDavid C Somayajulu u.wrf->buffer))) { 158f10a77bbSDavid C Somayajulu printf("flash write failed[%d]\n", rval); 159f10a77bbSDavid C Somayajulu rval = ENXIO; 160f10a77bbSDavid C Somayajulu } 161f10a77bbSDavid C Somayajulu break; 162f10a77bbSDavid C Somayajulu 163f10a77bbSDavid C Somayajulu case QLA_ERASE_FLASH: 164f10a77bbSDavid C Somayajulu 165f10a77bbSDavid C Somayajulu ifp = ha->ifp; 166f10a77bbSDavid C Somayajulu 167f10a77bbSDavid C Somayajulu if (ifp == NULL) { 168f10a77bbSDavid C Somayajulu rval = ENXIO; 169f10a77bbSDavid C Somayajulu break; 170f10a77bbSDavid C Somayajulu } 171f10a77bbSDavid C Somayajulu 172f10a77bbSDavid C Somayajulu if (ifp->if_drv_flags & (IFF_DRV_OACTIVE | IFF_DRV_RUNNING)) { 173f10a77bbSDavid C Somayajulu rval = ENXIO; 174f10a77bbSDavid C Somayajulu break; 175f10a77bbSDavid C Somayajulu } 176f10a77bbSDavid C Somayajulu 177f10a77bbSDavid C Somayajulu if (!ha->hw.flags.fdt_valid) { 178f10a77bbSDavid C Somayajulu rval = EIO; 179f10a77bbSDavid C Somayajulu break; 180f10a77bbSDavid C Somayajulu } 181f10a77bbSDavid C Somayajulu 182f10a77bbSDavid C Somayajulu u.erf = (qla_erase_flash_t *)data; 183f10a77bbSDavid C Somayajulu if ((rval = ql_erase_flash(ha, u.erf->off, 184f10a77bbSDavid C Somayajulu u.erf->size))) { 185f10a77bbSDavid C Somayajulu printf("flash erase failed[%d]\n", rval); 186f10a77bbSDavid C Somayajulu rval = ENXIO; 187f10a77bbSDavid C Somayajulu } 188f10a77bbSDavid C Somayajulu break; 189f10a77bbSDavid C Somayajulu 190f10a77bbSDavid C Somayajulu case QLA_RDWR_MS_MEM: 191f10a77bbSDavid C Somayajulu u.mem = (qla_offchip_mem_val_t *)data; 192f10a77bbSDavid C Somayajulu 193f10a77bbSDavid C Somayajulu if ((rval = ql_rdwr_offchip_mem(ha, u.mem->off, &val, 194f10a77bbSDavid C Somayajulu u.mem->rd))) 195f10a77bbSDavid C Somayajulu rval = ENXIO; 196f10a77bbSDavid C Somayajulu else { 197f10a77bbSDavid C Somayajulu u.mem->data_lo = val.data_lo; 198f10a77bbSDavid C Somayajulu u.mem->data_hi = val.data_hi; 199f10a77bbSDavid C Somayajulu u.mem->data_ulo = val.data_ulo; 200f10a77bbSDavid C Somayajulu u.mem->data_uhi = val.data_uhi; 201f10a77bbSDavid C Somayajulu } 202f10a77bbSDavid C Somayajulu 203f10a77bbSDavid C Somayajulu break; 204f10a77bbSDavid C Somayajulu 205f10a77bbSDavid C Somayajulu case QLA_RD_FW_DUMP_SIZE: 206f10a77bbSDavid C Somayajulu 207f10a77bbSDavid C Somayajulu if (ha->hw.mdump_init == 0) { 208f10a77bbSDavid C Somayajulu rval = EINVAL; 209f10a77bbSDavid C Somayajulu break; 210f10a77bbSDavid C Somayajulu } 211f10a77bbSDavid C Somayajulu 212f10a77bbSDavid C Somayajulu fw_dump = (qla_rd_fw_dump_t *)data; 213f10a77bbSDavid C Somayajulu fw_dump->template_size = ha->hw.dma_buf.minidump.size; 214f10a77bbSDavid C Somayajulu fw_dump->pci_func = ha->pci_func; 215f10a77bbSDavid C Somayajulu 216f10a77bbSDavid C Somayajulu break; 217f10a77bbSDavid C Somayajulu 218f10a77bbSDavid C Somayajulu case QLA_RD_FW_DUMP: 219f10a77bbSDavid C Somayajulu 220f10a77bbSDavid C Somayajulu if (ha->hw.mdump_init == 0) { 221f10a77bbSDavid C Somayajulu rval = EINVAL; 222f10a77bbSDavid C Somayajulu break; 223f10a77bbSDavid C Somayajulu } 224f10a77bbSDavid C Somayajulu 225f10a77bbSDavid C Somayajulu fw_dump = (qla_rd_fw_dump_t *)data; 226c12c5bfbSDavid C Somayajulu 227c12c5bfbSDavid C Somayajulu if ((fw_dump->md_template == NULL) || 228c12c5bfbSDavid C Somayajulu (fw_dump->template_size != ha->hw.dma_buf.minidump.size)) { 229c12c5bfbSDavid C Somayajulu rval = EINVAL; 230c12c5bfbSDavid C Somayajulu break; 231c12c5bfbSDavid C Somayajulu } 232c12c5bfbSDavid C Somayajulu 233f10a77bbSDavid C Somayajulu if ((rval = copyout(ha->hw.dma_buf.minidump.dma_b, 234f10a77bbSDavid C Somayajulu fw_dump->md_template, fw_dump->template_size))) 235f10a77bbSDavid C Somayajulu rval = ENXIO; 236f10a77bbSDavid C Somayajulu break; 237f10a77bbSDavid C Somayajulu 238f10a77bbSDavid C Somayajulu case QLA_RD_PCI_IDS: 239f10a77bbSDavid C Somayajulu pci_ids = (qla_rd_pci_ids_t *)data; 240f10a77bbSDavid C Somayajulu pci_ids->ven_id = pci_get_vendor(pci_dev); 241f10a77bbSDavid C Somayajulu pci_ids->dev_id = pci_get_device(pci_dev); 242f10a77bbSDavid C Somayajulu pci_ids->subsys_ven_id = pci_get_subvendor(pci_dev); 243f10a77bbSDavid C Somayajulu pci_ids->subsys_dev_id = pci_get_subdevice(pci_dev); 244f10a77bbSDavid C Somayajulu pci_ids->rev_id = pci_read_config(pci_dev, PCIR_REVID, 1); 245f10a77bbSDavid C Somayajulu break; 246f10a77bbSDavid C Somayajulu 247f10a77bbSDavid C Somayajulu default: 248f10a77bbSDavid C Somayajulu break; 249f10a77bbSDavid C Somayajulu } 250f10a77bbSDavid C Somayajulu 251f10a77bbSDavid C Somayajulu return rval; 252f10a77bbSDavid C Somayajulu } 253f10a77bbSDavid C Somayajulu 254