1*4882a593Smuzhiyun /*
2*4882a593Smuzhiyun * Linux driver attachment glue for aic7770 based controllers.
3*4882a593Smuzhiyun *
4*4882a593Smuzhiyun * Copyright (c) 2000-2003 Adaptec Inc.
5*4882a593Smuzhiyun * All rights reserved.
6*4882a593Smuzhiyun *
7*4882a593Smuzhiyun * Redistribution and use in source and binary forms, with or without
8*4882a593Smuzhiyun * modification, are permitted provided that the following conditions
9*4882a593Smuzhiyun * are met:
10*4882a593Smuzhiyun * 1. Redistributions of source code must retain the above copyright
11*4882a593Smuzhiyun * notice, this list of conditions, and the following disclaimer,
12*4882a593Smuzhiyun * without modification.
13*4882a593Smuzhiyun * 2. Redistributions in binary form must reproduce at minimum a disclaimer
14*4882a593Smuzhiyun * substantially similar to the "NO WARRANTY" disclaimer below
15*4882a593Smuzhiyun * ("Disclaimer") and any redistribution must be conditioned upon
16*4882a593Smuzhiyun * including a substantially similar Disclaimer requirement for further
17*4882a593Smuzhiyun * binary redistribution.
18*4882a593Smuzhiyun * 3. Neither the names of the above-listed copyright holders nor the names
19*4882a593Smuzhiyun * of any contributors may be used to endorse or promote products derived
20*4882a593Smuzhiyun * from this software without specific prior written permission.
21*4882a593Smuzhiyun *
22*4882a593Smuzhiyun * Alternatively, this software may be distributed under the terms of the
23*4882a593Smuzhiyun * GNU General Public License ("GPL") version 2 as published by the Free
24*4882a593Smuzhiyun * Software Foundation.
25*4882a593Smuzhiyun *
26*4882a593Smuzhiyun * NO WARRANTY
27*4882a593Smuzhiyun * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
28*4882a593Smuzhiyun * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
29*4882a593Smuzhiyun * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR
30*4882a593Smuzhiyun * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
31*4882a593Smuzhiyun * HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
32*4882a593Smuzhiyun * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
33*4882a593Smuzhiyun * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
34*4882a593Smuzhiyun * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
35*4882a593Smuzhiyun * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
36*4882a593Smuzhiyun * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
37*4882a593Smuzhiyun * POSSIBILITY OF SUCH DAMAGES.
38*4882a593Smuzhiyun *
39*4882a593Smuzhiyun * $Id: //depot/aic7xxx/linux/drivers/scsi/aic7xxx/aic7770_osm.c#14 $
40*4882a593Smuzhiyun */
41*4882a593Smuzhiyun
42*4882a593Smuzhiyun #include "aic7xxx_osm.h"
43*4882a593Smuzhiyun
44*4882a593Smuzhiyun #include <linux/device.h>
45*4882a593Smuzhiyun #include <linux/eisa.h>
46*4882a593Smuzhiyun
47*4882a593Smuzhiyun int
aic7770_map_registers(struct ahc_softc * ahc,u_int port)48*4882a593Smuzhiyun aic7770_map_registers(struct ahc_softc *ahc, u_int port)
49*4882a593Smuzhiyun {
50*4882a593Smuzhiyun /*
51*4882a593Smuzhiyun * Lock out other contenders for our i/o space.
52*4882a593Smuzhiyun */
53*4882a593Smuzhiyun if (!request_region(port, AHC_EISA_IOSIZE, "aic7xxx"))
54*4882a593Smuzhiyun return (ENOMEM);
55*4882a593Smuzhiyun ahc->tag = BUS_SPACE_PIO;
56*4882a593Smuzhiyun ahc->bsh.ioport = port;
57*4882a593Smuzhiyun return (0);
58*4882a593Smuzhiyun }
59*4882a593Smuzhiyun
60*4882a593Smuzhiyun int
aic7770_map_int(struct ahc_softc * ahc,u_int irq)61*4882a593Smuzhiyun aic7770_map_int(struct ahc_softc *ahc, u_int irq)
62*4882a593Smuzhiyun {
63*4882a593Smuzhiyun int error;
64*4882a593Smuzhiyun int shared;
65*4882a593Smuzhiyun
66*4882a593Smuzhiyun shared = 0;
67*4882a593Smuzhiyun if ((ahc->flags & AHC_EDGE_INTERRUPT) == 0)
68*4882a593Smuzhiyun shared = IRQF_SHARED;
69*4882a593Smuzhiyun
70*4882a593Smuzhiyun error = request_irq(irq, ahc_linux_isr, shared, "aic7xxx", ahc);
71*4882a593Smuzhiyun if (error == 0)
72*4882a593Smuzhiyun ahc->platform_data->irq = irq;
73*4882a593Smuzhiyun
74*4882a593Smuzhiyun return (-error);
75*4882a593Smuzhiyun }
76*4882a593Smuzhiyun
77*4882a593Smuzhiyun static int
aic7770_probe(struct device * dev)78*4882a593Smuzhiyun aic7770_probe(struct device *dev)
79*4882a593Smuzhiyun {
80*4882a593Smuzhiyun struct eisa_device *edev = to_eisa_device(dev);
81*4882a593Smuzhiyun u_int eisaBase = edev->base_addr+AHC_EISA_SLOT_OFFSET;
82*4882a593Smuzhiyun struct ahc_softc *ahc;
83*4882a593Smuzhiyun char buf[80];
84*4882a593Smuzhiyun char *name;
85*4882a593Smuzhiyun int error;
86*4882a593Smuzhiyun
87*4882a593Smuzhiyun sprintf(buf, "ahc_eisa:%d", eisaBase >> 12);
88*4882a593Smuzhiyun name = kstrdup(buf, GFP_ATOMIC);
89*4882a593Smuzhiyun if (name == NULL)
90*4882a593Smuzhiyun return (ENOMEM);
91*4882a593Smuzhiyun ahc = ahc_alloc(&aic7xxx_driver_template, name);
92*4882a593Smuzhiyun if (ahc == NULL)
93*4882a593Smuzhiyun return (ENOMEM);
94*4882a593Smuzhiyun ahc->dev = dev;
95*4882a593Smuzhiyun error = aic7770_config(ahc, aic7770_ident_table + edev->id.driver_data,
96*4882a593Smuzhiyun eisaBase);
97*4882a593Smuzhiyun if (error != 0) {
98*4882a593Smuzhiyun ahc->bsh.ioport = 0;
99*4882a593Smuzhiyun ahc_free(ahc);
100*4882a593Smuzhiyun return (error);
101*4882a593Smuzhiyun }
102*4882a593Smuzhiyun
103*4882a593Smuzhiyun dev_set_drvdata(dev, ahc);
104*4882a593Smuzhiyun
105*4882a593Smuzhiyun error = ahc_linux_register_host(ahc, &aic7xxx_driver_template);
106*4882a593Smuzhiyun return (error);
107*4882a593Smuzhiyun }
108*4882a593Smuzhiyun
109*4882a593Smuzhiyun static int
aic7770_remove(struct device * dev)110*4882a593Smuzhiyun aic7770_remove(struct device *dev)
111*4882a593Smuzhiyun {
112*4882a593Smuzhiyun struct ahc_softc *ahc = dev_get_drvdata(dev);
113*4882a593Smuzhiyun u_long s;
114*4882a593Smuzhiyun
115*4882a593Smuzhiyun if (ahc->platform_data && ahc->platform_data->host)
116*4882a593Smuzhiyun scsi_remove_host(ahc->platform_data->host);
117*4882a593Smuzhiyun
118*4882a593Smuzhiyun ahc_lock(ahc, &s);
119*4882a593Smuzhiyun ahc_intr_enable(ahc, FALSE);
120*4882a593Smuzhiyun ahc_unlock(ahc, &s);
121*4882a593Smuzhiyun
122*4882a593Smuzhiyun ahc_free(ahc);
123*4882a593Smuzhiyun return 0;
124*4882a593Smuzhiyun }
125*4882a593Smuzhiyun
126*4882a593Smuzhiyun static struct eisa_device_id aic7770_ids[] = {
127*4882a593Smuzhiyun { "ADP7771", 0 }, /* AHA 274x */
128*4882a593Smuzhiyun { "ADP7756", 1 }, /* AHA 284x BIOS enabled */
129*4882a593Smuzhiyun { "ADP7757", 2 }, /* AHA 284x BIOS disabled */
130*4882a593Smuzhiyun { "ADP7782", 3 }, /* AHA 274x Olivetti OEM */
131*4882a593Smuzhiyun { "ADP7783", 4 }, /* AHA 274x Olivetti OEM (Differential) */
132*4882a593Smuzhiyun { "ADP7770", 5 }, /* AIC7770 generic */
133*4882a593Smuzhiyun { "" }
134*4882a593Smuzhiyun };
135*4882a593Smuzhiyun MODULE_DEVICE_TABLE(eisa, aic7770_ids);
136*4882a593Smuzhiyun
137*4882a593Smuzhiyun static struct eisa_driver aic7770_driver = {
138*4882a593Smuzhiyun .id_table = aic7770_ids,
139*4882a593Smuzhiyun .driver = {
140*4882a593Smuzhiyun .name = "aic7xxx",
141*4882a593Smuzhiyun .probe = aic7770_probe,
142*4882a593Smuzhiyun .remove = aic7770_remove,
143*4882a593Smuzhiyun }
144*4882a593Smuzhiyun };
145*4882a593Smuzhiyun
146*4882a593Smuzhiyun int
ahc_linux_eisa_init(void)147*4882a593Smuzhiyun ahc_linux_eisa_init(void)
148*4882a593Smuzhiyun {
149*4882a593Smuzhiyun return eisa_driver_register(&aic7770_driver);
150*4882a593Smuzhiyun }
151*4882a593Smuzhiyun
152*4882a593Smuzhiyun void
ahc_linux_eisa_exit(void)153*4882a593Smuzhiyun ahc_linux_eisa_exit(void)
154*4882a593Smuzhiyun {
155*4882a593Smuzhiyun eisa_driver_unregister(&aic7770_driver);
156*4882a593Smuzhiyun }
157