}
 #endif /* SYM_LINUX_PROC_INFO_SUPPORT */
 
+/*
+ * Free resources claimed by sym_init_device().  Note that
+ * sym_free_resources() should be used instead of this function after calling
+ * sym_attach().
+ */
+static void __devinit
+sym_iounmap_device(struct sym_device *device)
+{
+       if (device->s.ioaddr)
+               pci_iounmap(device->pdev, device->s.ioaddr);
+       if (device->s.ramaddr)
+               pci_iounmap(device->pdev, device->s.ramaddr);
+}
+
 /*
  *     Free controller resources.
  */
 {
        struct sym_data *sym_data;
        struct sym_hcb *np = NULL;
-       struct Scsi_Host *shost;
+       struct Scsi_Host *shost = NULL;
        struct pci_dev *pdev = dev->pdev;
        unsigned long flags;
        struct sym_fw *fw;
         */
        fw = sym_find_firmware(&dev->chip);
        if (!fw)
-               return NULL;
+               goto attach_failed;
 
        shost = scsi_host_alloc(tpnt, sizeof(*sym_data));
        if (!shost)
-               return NULL;
+               goto attach_failed;
        sym_data = shost_priv(shost);
 
        /*
        np->maxoffs     = dev->chip.offset_max;
        np->maxburst    = dev->chip.burst_max;
        np->myaddr      = dev->host_id;
+       np->mmio_ba     = (u32)dev->mmio_base;
+       np->s.ioaddr    = dev->s.ioaddr;
+       np->s.ramaddr   = dev->s.ramaddr;
+       if (!(np->features & FE_RAM))
+               dev->ram_base = 0;
+       if (dev->ram_base)
+               np->ram_ba = (u32)dev->ram_base;
 
        /*
         *  Edit its name.
                goto attach_failed;
        }
 
-       /*
-        *  Try to map the controller chip to
-        *  virtual and physical memory.
-        */
-       np->mmio_ba = (u32)dev->mmio_base;
-       np->s.ioaddr    = dev->s.ioaddr;
-       np->s.ramaddr   = dev->s.ramaddr;
-
-       /*
-        *  Map on-chip RAM if present and supported.
-        */
-       if (!(np->features & FE_RAM))
-               dev->ram_base = 0;
-       if (dev->ram_base)
-               np->ram_ba = (u32)dev->ram_base;
-
        if (sym_hcb_attach(shost, fw, dev->nvram))
                goto attach_failed;
 
                   "TERMINATION, DEVICE POWER etc.!\n", sym_name(np));
        spin_unlock_irqrestore(shost->host_lock, flags);
  attach_failed:
-       if (!shost)
-               return NULL;
        printf_info("sym%d: giving up ...\n", unit);
        if (np)
                sym_free_resources(np, pdev, do_free_irq);
-       scsi_host_put(shost);
+       else
+               sym_iounmap_device(dev);
+       if (shost)
+               scsi_host_put(shost);
 
        return NULL;
  }
        struct sym_device sym_dev;
        struct sym_nvram nvram;
        struct Scsi_Host *shost;
+       int do_iounmap = 0;
+       int do_disable_device = 1;
 
        memset(&sym_dev, 0, sizeof(sym_dev));
        memset(&nvram, 0, sizeof(nvram));
                goto disable;
 
        sym_init_device(pdev, &sym_dev);
+       do_iounmap = 1;
+
        if (sym_check_supported(&sym_dev))
                goto free;
 
-       if (sym_check_raid(&sym_dev))
-               goto leave;     /* Don't disable the device */
+       if (sym_check_raid(&sym_dev)) {
+               do_disable_device = 0;  /* Don't disable the device */
+               goto free;
+       }
 
        if (sym_set_workarounds(&sym_dev))
                goto free;
 
        sym_get_nvram(&sym_dev, &nvram);
 
+       do_iounmap = 0; /* Don't sym_iounmap_device() after sym_attach(). */
        shost = sym_attach(&sym2_template, attach_count, &sym_dev);
        if (!shost)
                goto free;
  detach:
        sym_detach(pci_get_drvdata(pdev), pdev);
  free:
+       if (do_iounmap)
+               sym_iounmap_device(&sym_dev);
        pci_release_regions(pdev);
  disable:
-       pci_disable_device(pdev);
+       if (do_disable_device)
+               pci_disable_device(pdev);
  leave:
        return -ENODEV;
 }