| /* |
| * Disk Array driver for Compaq SMART2 Controllers |
| * Copyright 1998 Compaq Computer Corporation |
| * |
| * This program is free software; you can redistribute it and/or modify |
| * it under the terms of the GNU General Public License as published by |
| * the Free Software Foundation; either version 2 of the License, or |
| * (at your option) any later version. |
| * |
| * This program is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE, GOOD TITLE or |
| * NON INFRINGEMENT. See the GNU General Public License for more details. |
| * |
| * You should have received a copy of the GNU General Public License |
| * along with this program; if not, write to the Free Software |
| * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. |
| * |
| * Questions/Comments/Bugfixes to iss_storagedev@hp.com |
| * |
| * If you want to make changes, improve or add functionality to this |
| * driver, you'll probably need the Compaq Array Controller Interface |
| * Specificiation (Document number ECG086/1198) |
| */ |
| |
| /* |
| * This file contains the controller communication implementation for |
| * Compaq SMART-1 and SMART-2 controllers. To the best of my knowledge, |
| * this should support: |
| * |
| * PCI: |
| * SMART-2/P, SMART-2DH, SMART-2SL, SMART-221, SMART-3100ES, SMART-3200 |
| * Integerated SMART Array Controller, SMART-4200, SMART-4250ES |
| * |
| * EISA: |
| * SMART-2/E, SMART, IAES, IDA-2, IDA |
| */ |
| |
| /* |
| * Memory mapped FIFO interface (SMART 42xx cards) |
| */ |
| static void smart4_submit_command(ctlr_info_t *h, cmdlist_t *c) |
| { |
| writel(c->busaddr, h->vaddr + S42XX_REQUEST_PORT_OFFSET); |
| } |
| |
| /* |
| * This card is the opposite of the other cards. |
| * 0 turns interrupts on... |
| * 0x08 turns them off... |
| */ |
| static void smart4_intr_mask(ctlr_info_t *h, unsigned long val) |
| { |
| if (val) |
| { /* Turn interrupts on */ |
| writel(0, h->vaddr + S42XX_REPLY_INTR_MASK_OFFSET); |
| } else /* Turn them off */ |
| { |
| writel( S42XX_INTR_OFF, |
| h->vaddr + S42XX_REPLY_INTR_MASK_OFFSET); |
| } |
| } |
| |
| /* |
| * For older cards FIFO Full = 0. |
| * On this card 0 means there is room, anything else FIFO Full. |
| * |
| */ |
| static unsigned long smart4_fifo_full(ctlr_info_t *h) |
| { |
| |
| return (!readl(h->vaddr + S42XX_REQUEST_PORT_OFFSET)); |
| } |
| |
| /* This type of controller returns -1 if the fifo is empty, |
| * Not 0 like the others. |
| * And we need to let it know we read a value out |
| */ |
| static unsigned long smart4_completed(ctlr_info_t *h) |
| { |
| long register_value |
| = readl(h->vaddr + S42XX_REPLY_PORT_OFFSET); |
| |
| /* Fifo is empty */ |
| if( register_value == 0xffffffff) |
| return 0; |
| |
| /* Need to let it know we got the reply */ |
| /* We do this by writing a 0 to the port we just read from */ |
| writel(0, h->vaddr + S42XX_REPLY_PORT_OFFSET); |
| |
| return ((unsigned long) register_value); |
| } |
| |
| /* |
| * This hardware returns interrupt pending at a different place and |
| * it does not tell us if the fifo is empty, we will have check |
| * that by getting a 0 back from the command_completed call. |
| */ |
| static unsigned long smart4_intr_pending(ctlr_info_t *h) |
| { |
| unsigned long register_value = |
| readl(h->vaddr + S42XX_INTR_STATUS); |
| |
| if( register_value & S42XX_INTR_PENDING) |
| return FIFO_NOT_EMPTY; |
| return 0 ; |
| } |
| |
| static struct access_method smart4_access = { |
| smart4_submit_command, |
| smart4_intr_mask, |
| smart4_fifo_full, |
| smart4_intr_pending, |
| smart4_completed, |
| }; |
| |
| /* |
| * Memory mapped FIFO interface (PCI SMART2 and SMART 3xxx cards) |
| */ |
| static void smart2_submit_command(ctlr_info_t *h, cmdlist_t *c) |
| { |
| writel(c->busaddr, h->vaddr + COMMAND_FIFO); |
| } |
| |
| static void smart2_intr_mask(ctlr_info_t *h, unsigned long val) |
| { |
| writel(val, h->vaddr + INTR_MASK); |
| } |
| |
| static unsigned long smart2_fifo_full(ctlr_info_t *h) |
| { |
| return readl(h->vaddr + COMMAND_FIFO); |
| } |
| |
| static unsigned long smart2_completed(ctlr_info_t *h) |
| { |
| return readl(h->vaddr + COMMAND_COMPLETE_FIFO); |
| } |
| |
| static unsigned long smart2_intr_pending(ctlr_info_t *h) |
| { |
| return readl(h->vaddr + INTR_PENDING); |
| } |
| |
| static struct access_method smart2_access = { |
| smart2_submit_command, |
| smart2_intr_mask, |
| smart2_fifo_full, |
| smart2_intr_pending, |
| smart2_completed, |
| }; |
| |
| /* |
| * IO access for SMART-2/E cards |
| */ |
| static void smart2e_submit_command(ctlr_info_t *h, cmdlist_t *c) |
| { |
| outl(c->busaddr, h->io_mem_addr + COMMAND_FIFO); |
| } |
| |
| static void smart2e_intr_mask(ctlr_info_t *h, unsigned long val) |
| { |
| outl(val, h->io_mem_addr + INTR_MASK); |
| } |
| |
| static unsigned long smart2e_fifo_full(ctlr_info_t *h) |
| { |
| return inl(h->io_mem_addr + COMMAND_FIFO); |
| } |
| |
| static unsigned long smart2e_completed(ctlr_info_t *h) |
| { |
| return inl(h->io_mem_addr + COMMAND_COMPLETE_FIFO); |
| } |
| |
| static unsigned long smart2e_intr_pending(ctlr_info_t *h) |
| { |
| return inl(h->io_mem_addr + INTR_PENDING); |
| } |
| |
| static struct access_method smart2e_access = { |
| smart2e_submit_command, |
| smart2e_intr_mask, |
| smart2e_fifo_full, |
| smart2e_intr_pending, |
| smart2e_completed, |
| }; |
| |
| /* |
| * IO access for older SMART-1 type cards |
| */ |
| #define SMART1_SYSTEM_MASK 0xC8E |
| #define SMART1_SYSTEM_DOORBELL 0xC8F |
| #define SMART1_LOCAL_MASK 0xC8C |
| #define SMART1_LOCAL_DOORBELL 0xC8D |
| #define SMART1_INTR_MASK 0xC89 |
| #define SMART1_LISTADDR 0xC90 |
| #define SMART1_LISTLEN 0xC94 |
| #define SMART1_TAG 0xC97 |
| #define SMART1_COMPLETE_ADDR 0xC98 |
| #define SMART1_LISTSTATUS 0xC9E |
| |
| #define CHANNEL_BUSY 0x01 |
| #define CHANNEL_CLEAR 0x02 |
| |
| static void smart1_submit_command(ctlr_info_t *h, cmdlist_t *c) |
| { |
| /* |
| * This __u16 is actually a bunch of control flags on SMART |
| * and below. We want them all to be zero. |
| */ |
| c->hdr.size = 0; |
| |
| outb(CHANNEL_CLEAR, h->io_mem_addr + SMART1_SYSTEM_DOORBELL); |
| |
| outl(c->busaddr, h->io_mem_addr + SMART1_LISTADDR); |
| outw(c->size, h->io_mem_addr + SMART1_LISTLEN); |
| |
| outb(CHANNEL_BUSY, h->io_mem_addr + SMART1_LOCAL_DOORBELL); |
| } |
| |
| static void smart1_intr_mask(ctlr_info_t *h, unsigned long val) |
| { |
| if (val == 1) { |
| outb(0xFD, h->io_mem_addr + SMART1_SYSTEM_DOORBELL); |
| outb(CHANNEL_BUSY, h->io_mem_addr + SMART1_LOCAL_DOORBELL); |
| outb(0x01, h->io_mem_addr + SMART1_INTR_MASK); |
| outb(0x01, h->io_mem_addr + SMART1_SYSTEM_MASK); |
| } else { |
| outb(0, h->io_mem_addr + 0xC8E); |
| } |
| } |
| |
| static unsigned long smart1_fifo_full(ctlr_info_t *h) |
| { |
| unsigned char chan; |
| chan = inb(h->io_mem_addr + SMART1_SYSTEM_DOORBELL) & CHANNEL_CLEAR; |
| return chan; |
| } |
| |
| static unsigned long smart1_completed(ctlr_info_t *h) |
| { |
| unsigned char status; |
| unsigned long cmd; |
| |
| if (inb(h->io_mem_addr + SMART1_SYSTEM_DOORBELL) & CHANNEL_BUSY) { |
| outb(CHANNEL_BUSY, h->io_mem_addr + SMART1_SYSTEM_DOORBELL); |
| |
| cmd = inl(h->io_mem_addr + SMART1_COMPLETE_ADDR); |
| status = inb(h->io_mem_addr + SMART1_LISTSTATUS); |
| |
| outb(CHANNEL_CLEAR, h->io_mem_addr + SMART1_LOCAL_DOORBELL); |
| |
| /* |
| * this is x86 (actually compaq x86) only, so it's ok |
| */ |
| if (cmd) ((cmdlist_t*)bus_to_virt(cmd))->req.hdr.rcode = status; |
| } else { |
| cmd = 0; |
| } |
| return cmd; |
| } |
| |
| static unsigned long smart1_intr_pending(ctlr_info_t *h) |
| { |
| unsigned char chan; |
| chan = inb(h->io_mem_addr + SMART1_SYSTEM_DOORBELL) & CHANNEL_BUSY; |
| return chan; |
| } |
| |
| static struct access_method smart1_access = { |
| smart1_submit_command, |
| smart1_intr_mask, |
| smart1_fifo_full, |
| smart1_intr_pending, |
| smart1_completed, |
| }; |