提交 551e4fb2 编写于 作者: L Linus Torvalds

Merge git://git.kernel.org/pub/scm/linux/kernel/git/bart/ide-2.6

* git://git.kernel.org/pub/scm/linux/kernel/git/bart/ide-2.6: (29 commits)
  ide-tape: bump minor driver version
  ide-tape: cleanup the remaining codestyle issues
  ide-tape: fix syntax error in idetape_identify_device()
  ide-tape: remove leftover OnStream support warning
  ide-tape: collect module-related macro calls at the end
  ide-tape: include proper headers
  ide-tape: remove unused "length" arg from idetape_create_read_buffer_cmd()
  ide-tape: remove struct idetape_id_gcw
  ide-tape: cleanup and fix comments
  ide-tape: shorten some function names
  ide-tape: remove idetape_increase_max_pipeline_stages()
  ide-tape: struct idetape_tape_t: shorten member names v2
  ide-tape: struct idetape_tape_t: remove unused members
  ide-tape: remove typedef idetape_chrdev_direction_t
  ide-tape: simplify code branching in the interrupt handler
  ide-tape: remove unreachable code chunk
  ide-tape: remove struct idetape_read_position_result_t
  ide-tape: refactor the debug logging facility
  ide: add ide_read_error() inline helper
  ide: add ide_read_[alt]status() inline helpers
  ...
...@@ -780,6 +780,9 @@ and is between 256 and 4096 characters. It is defined in the file ...@@ -780,6 +780,9 @@ and is between 256 and 4096 characters. It is defined in the file
loop use the MONITOR/MWAIT idle loop anyways. Performance should be the same loop use the MONITOR/MWAIT idle loop anyways. Performance should be the same
as idle=poll. as idle=poll.
ide-pci-generic.all-generic-ide [HW] (E)IDE subsystem
Claim all unknown PCI IDE storage controllers.
ignore_loglevel [KNL] ignore_loglevel [KNL]
Ignore loglevel setting - this will print /all/ Ignore loglevel setting - this will print /all/
kernel messages to the console. Useful for debugging. kernel messages to the console. Useful for debugging.
......
...@@ -1009,6 +1009,15 @@ config BLK_DEV_Q40IDE ...@@ -1009,6 +1009,15 @@ config BLK_DEV_Q40IDE
normally be on; disable it only if you are running a custom hard normally be on; disable it only if you are running a custom hard
drive subsystem through an expansion card. drive subsystem through an expansion card.
config BLK_DEV_PALMCHIP_BK3710
tristate "Palmchip bk3710 IDE controller support"
depends on ARCH_DAVINCI
select BLK_DEV_IDEDMA_PCI
help
Say Y here if you want to support the onchip IDE controller on the
TI DaVinci SoC
config BLK_DEV_MPC8xx_IDE config BLK_DEV_MPC8xx_IDE
tristate "MPC8xx IDE support" tristate "MPC8xx IDE support"
depends on 8xx && (LWMON || IVMS8 || IVML24 || TQM8xxL) && IDE=y && BLK_DEV_IDE=y && !PPC_MERGE depends on 8xx && (LWMON || IVMS8 || IVML24 || TQM8xxL) && IDE=y && BLK_DEV_IDE=y && !PPC_MERGE
......
...@@ -2,6 +2,7 @@ ...@@ -2,6 +2,7 @@
obj-$(CONFIG_BLK_DEV_IDE_ICSIDE) += icside.o obj-$(CONFIG_BLK_DEV_IDE_ICSIDE) += icside.o
obj-$(CONFIG_BLK_DEV_IDE_RAPIDE) += rapide.o obj-$(CONFIG_BLK_DEV_IDE_RAPIDE) += rapide.o
obj-$(CONFIG_BLK_DEV_IDE_BAST) += bast-ide.o obj-$(CONFIG_BLK_DEV_IDE_BAST) += bast-ide.o
obj-$(CONFIG_BLK_DEV_PALMCHIP_BK3710) += palm_bk3710.o
ifeq ($(CONFIG_IDE_ARM), m) ifeq ($(CONFIG_IDE_ARM), m)
obj-m += ide_arm.o obj-m += ide_arm.o
......
...@@ -365,7 +365,7 @@ static void icside_dma_timeout(ide_drive_t *drive) ...@@ -365,7 +365,7 @@ static void icside_dma_timeout(ide_drive_t *drive)
if (icside_dma_test_irq(drive)) if (icside_dma_test_irq(drive))
return; return;
ide_dump_status(drive, "DMA timeout", HWIF(drive)->INB(IDE_STATUS_REG)); ide_dump_status(drive, "DMA timeout", ide_read_status(drive));
icside_dma_end(drive); icside_dma_end(drive);
} }
......
/*
* Palmchip bk3710 IDE controller
*
* Copyright (C) 2006 Texas Instruments.
* Copyright (C) 2007 MontaVista Software, Inc., <source@mvista.com>
*
* ----------------------------------------------------------------------------
*
* 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. 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.
* ----------------------------------------------------------------------------
*
*/
#include <linux/types.h>
#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/ioport.h>
#include <linux/hdreg.h>
#include <linux/ide.h>
#include <linux/delay.h>
#include <linux/init.h>
#include <linux/clk.h>
#include <linux/platform_device.h>
/* Offset of the primary interface registers */
#define IDE_PALM_ATA_PRI_REG_OFFSET 0x1F0
/* Primary Control Offset */
#define IDE_PALM_ATA_PRI_CTL_OFFSET 0x3F6
/*
* PalmChip 3710 IDE Controller UDMA timing structure Definition
*/
struct palm_bk3710_udmatiming {
unsigned int rptime; /* Ready to pause time */
unsigned int cycletime; /* Cycle Time */
};
#define BK3710_BMICP 0x00
#define BK3710_BMISP 0x02
#define BK3710_BMIDTP 0x04
#define BK3710_BMICS 0x08
#define BK3710_BMISS 0x0A
#define BK3710_BMIDTS 0x0C
#define BK3710_IDETIMP 0x40
#define BK3710_IDETIMS 0x42
#define BK3710_SIDETIM 0x44
#define BK3710_SLEWCTL 0x45
#define BK3710_IDESTATUS 0x47
#define BK3710_UDMACTL 0x48
#define BK3710_UDMATIM 0x4A
#define BK3710_MISCCTL 0x50
#define BK3710_REGSTB 0x54
#define BK3710_REGRCVR 0x58
#define BK3710_DATSTB 0x5C
#define BK3710_DATRCVR 0x60
#define BK3710_DMASTB 0x64
#define BK3710_DMARCVR 0x68
#define BK3710_UDMASTB 0x6C
#define BK3710_UDMATRP 0x70
#define BK3710_UDMAENV 0x74
#define BK3710_IORDYTMP 0x78
#define BK3710_IORDYTMS 0x7C
#include "../ide-timing.h"
static long ide_palm_clk;
static const struct palm_bk3710_udmatiming palm_bk3710_udmatimings[6] = {
{160, 240}, /* UDMA Mode 0 */
{125, 160}, /* UDMA Mode 1 */
{100, 120}, /* UDMA Mode 2 */
{100, 90}, /* UDMA Mode 3 */
{85, 60}, /* UDMA Mode 4 */
};
static struct clk *ideclkp;
static void palm_bk3710_setudmamode(void __iomem *base, unsigned int dev,
unsigned int mode)
{
u8 tenv, trp, t0;
u32 val32;
u16 val16;
/* DMA Data Setup */
t0 = (palm_bk3710_udmatimings[mode].cycletime + ide_palm_clk - 1)
/ ide_palm_clk - 1;
tenv = (20 + ide_palm_clk - 1) / ide_palm_clk - 1;
trp = (palm_bk3710_udmatimings[mode].rptime + ide_palm_clk - 1)
/ ide_palm_clk - 1;
/* udmatim Register */
val16 = readw(base + BK3710_UDMATIM) & (dev ? 0xFF0F : 0xFFF0);
val16 |= (mode << (dev ? 4 : 0));
writew(val16, base + BK3710_UDMATIM);
/* udmastb Ultra DMA Access Strobe Width */
val32 = readl(base + BK3710_UDMASTB) & (0xFF << (dev ? 0 : 8));
val32 |= (t0 << (dev ? 8 : 0));
writel(val32, base + BK3710_UDMASTB);
/* udmatrp Ultra DMA Ready to Pause Time */
val32 = readl(base + BK3710_UDMATRP) & (0xFF << (dev ? 0 : 8));
val32 |= (trp << (dev ? 8 : 0));
writel(val32, base + BK3710_UDMATRP);
/* udmaenv Ultra DMA envelop Time */
val32 = readl(base + BK3710_UDMAENV) & (0xFF << (dev ? 0 : 8));
val32 |= (tenv << (dev ? 8 : 0));
writel(val32, base + BK3710_UDMAENV);
/* Enable UDMA for Device */
val16 = readw(base + BK3710_UDMACTL) | (1 << dev);
writew(val16, base + BK3710_UDMACTL);
}
static void palm_bk3710_setdmamode(void __iomem *base, unsigned int dev,
unsigned short min_cycle,
unsigned int mode)
{
u8 td, tkw, t0;
u32 val32;
u16 val16;
struct ide_timing *t;
int cycletime;
t = ide_timing_find_mode(mode);
cycletime = max_t(int, t->cycle, min_cycle);
/* DMA Data Setup */
t0 = (cycletime + ide_palm_clk - 1) / ide_palm_clk;
td = (t->active + ide_palm_clk - 1) / ide_palm_clk;
tkw = t0 - td - 1;
td -= 1;
val32 = readl(base + BK3710_DMASTB) & (0xFF << (dev ? 0 : 8));
val32 |= (td << (dev ? 8 : 0));
writel(val32, base + BK3710_DMASTB);
val32 = readl(base + BK3710_DMARCVR) & (0xFF << (dev ? 0 : 8));
val32 |= (tkw << (dev ? 8 : 0));
writel(val32, base + BK3710_DMARCVR);
/* Disable UDMA for Device */
val16 = readw(base + BK3710_UDMACTL) & ~(1 << dev);
writew(val16, base + BK3710_UDMACTL);
}
static void palm_bk3710_setpiomode(void __iomem *base, ide_drive_t *mate,
unsigned int dev, unsigned int cycletime,
unsigned int mode)
{
u8 t2, t2i, t0;
u32 val32;
struct ide_timing *t;
/* PIO Data Setup */
t0 = (cycletime + ide_palm_clk - 1) / ide_palm_clk;
t2 = (ide_timing_find_mode(XFER_PIO_0 + mode)->active +
ide_palm_clk - 1) / ide_palm_clk;
t2i = t0 - t2 - 1;
t2 -= 1;
val32 = readl(base + BK3710_DATSTB) & (0xFF << (dev ? 0 : 8));
val32 |= (t2 << (dev ? 8 : 0));
writel(val32, base + BK3710_DATSTB);
val32 = readl(base + BK3710_DATRCVR) & (0xFF << (dev ? 0 : 8));
val32 |= (t2i << (dev ? 8 : 0));
writel(val32, base + BK3710_DATRCVR);
if (mate && mate->present) {
u8 mode2 = ide_get_best_pio_mode(mate, 255, 4);
if (mode2 < mode)
mode = mode2;
}
/* TASKFILE Setup */
t = ide_timing_find_mode(XFER_PIO_0 + mode);
t0 = (t->cyc8b + ide_palm_clk - 1) / ide_palm_clk;
t2 = (t->act8b + ide_palm_clk - 1) / ide_palm_clk;
t2i = t0 - t2 - 1;
t2 -= 1;
val32 = readl(base + BK3710_REGSTB) & (0xFF << (dev ? 0 : 8));
val32 |= (t2 << (dev ? 8 : 0));
writel(val32, base + BK3710_REGSTB);
val32 = readl(base + BK3710_REGRCVR) & (0xFF << (dev ? 0 : 8));
val32 |= (t2i << (dev ? 8 : 0));
writel(val32, base + BK3710_REGRCVR);
}
static void palm_bk3710_set_dma_mode(ide_drive_t *drive, u8 xferspeed)
{
int is_slave = drive->dn & 1;
void __iomem *base = (void *)drive->hwif->dma_base;
if (xferspeed >= XFER_UDMA_0) {
palm_bk3710_setudmamode(base, is_slave,
xferspeed - XFER_UDMA_0);
} else {
palm_bk3710_setdmamode(base, is_slave, drive->id->eide_dma_min,
xferspeed);
}
}
static void palm_bk3710_set_pio_mode(ide_drive_t *drive, u8 pio)
{
unsigned int cycle_time;
int is_slave = drive->dn & 1;
ide_drive_t *mate;
void __iomem *base = (void *)drive->hwif->dma_base;
/*
* Obtain the drive PIO data for tuning the Palm Chip registers
*/
cycle_time = ide_pio_cycle_time(drive, pio);
mate = ide_get_paired_drive(drive);
palm_bk3710_setpiomode(base, mate, is_slave, cycle_time, pio);
}
static void __devinit palm_bk3710_chipinit(void __iomem *base)
{
/*
* enable the reset_en of ATA controller so that when ata signals
* are brought out, by writing into device config. at that
* time por_n signal should not be 'Z' and have a stable value.
*/
writel(0x0300, base + BK3710_MISCCTL);
/* wait for some time and deassert the reset of ATA Device. */
mdelay(100);
/* Deassert the Reset */
writel(0x0200, base + BK3710_MISCCTL);
/*
* Program the IDETIMP Register Value based on the following assumptions
*
* (ATA_IDETIMP_IDEEN , ENABLE ) |
* (ATA_IDETIMP_SLVTIMEN , DISABLE) |
* (ATA_IDETIMP_RDYSMPL , 70NS) |
* (ATA_IDETIMP_RDYRCVRY , 50NS) |
* (ATA_IDETIMP_DMAFTIM1 , PIOCOMP) |
* (ATA_IDETIMP_PREPOST1 , DISABLE) |
* (ATA_IDETIMP_RDYSEN1 , DISABLE) |
* (ATA_IDETIMP_PIOFTIM1 , DISABLE) |
* (ATA_IDETIMP_DMAFTIM0 , PIOCOMP) |
* (ATA_IDETIMP_PREPOST0 , DISABLE) |
* (ATA_IDETIMP_RDYSEN0 , DISABLE) |
* (ATA_IDETIMP_PIOFTIM0 , DISABLE)
*/
writew(0xB388, base + BK3710_IDETIMP);
/*
* Configure SIDETIM Register
* (ATA_SIDETIM_RDYSMPS1 ,120NS ) |
* (ATA_SIDETIM_RDYRCYS1 ,120NS )
*/
writeb(0, base + BK3710_SIDETIM);
/*
* UDMACTL Ultra-ATA DMA Control
* (ATA_UDMACTL_UDMAP1 , 0 ) |
* (ATA_UDMACTL_UDMAP0 , 0 )
*
*/
writew(0, base + BK3710_UDMACTL);
/*
* MISCCTL Miscellaneous Conrol Register
* (ATA_MISCCTL_RSTMODEP , 1) |
* (ATA_MISCCTL_RESETP , 0) |
* (ATA_MISCCTL_TIMORIDE , 1)
*/
writel(0x201, base + BK3710_MISCCTL);
/*
* IORDYTMP IORDY Timer for Primary Register
* (ATA_IORDYTMP_IORDYTMP , 0xffff )
*/
writel(0xFFFF, base + BK3710_IORDYTMP);
/*
* Configure BMISP Register
* (ATA_BMISP_DMAEN1 , DISABLE ) |
* (ATA_BMISP_DMAEN0 , DISABLE ) |
* (ATA_BMISP_IORDYINT , CLEAR) |
* (ATA_BMISP_INTRSTAT , CLEAR) |
* (ATA_BMISP_DMAERROR , CLEAR)
*/
writew(0, base + BK3710_BMISP);
palm_bk3710_setpiomode(base, NULL, 0, 600, 0);
palm_bk3710_setpiomode(base, NULL, 1, 600, 0);
}
static int __devinit palm_bk3710_probe(struct platform_device *pdev)
{
hw_regs_t ide_ctlr_info;
int index = 0;
int pribase;
struct clk *clkp;
struct resource *mem, *irq;
ide_hwif_t *hwif;
void __iomem *base;
clkp = clk_get(NULL, "IDECLK");
if (IS_ERR(clkp))
return -ENODEV;
ideclkp = clkp;
clk_enable(ideclkp);
ide_palm_clk = clk_get_rate(ideclkp)/100000;
ide_palm_clk = (10000/ide_palm_clk) + 1;
/* Register the IDE interface with Linux ATA Interface */
memset(&ide_ctlr_info, 0, sizeof(ide_ctlr_info));
mem = platform_get_resource(pdev, IORESOURCE_MEM, 0);
if (mem == NULL) {
printk(KERN_ERR "failed to get memory region resource\n");
return -ENODEV;
}
irq = platform_get_resource(pdev, IORESOURCE_IRQ, 0);
if (irq == NULL) {
printk(KERN_ERR "failed to get IRQ resource\n");
return -ENODEV;
}
base = (void *)mem->start;
/* Configure the Palm Chip controller */
palm_bk3710_chipinit(base);
pribase = mem->start + IDE_PALM_ATA_PRI_REG_OFFSET;
for (index = 0; index < IDE_NR_PORTS - 2; index++)
ide_ctlr_info.io_ports[index] = pribase + index;
ide_ctlr_info.io_ports[IDE_CONTROL_OFFSET] = mem->start +
IDE_PALM_ATA_PRI_CTL_OFFSET;
ide_ctlr_info.irq = irq->start;
ide_ctlr_info.chipset = ide_palm3710;
if (ide_register_hw(&ide_ctlr_info, NULL, &hwif) < 0) {
printk(KERN_WARNING "Palm Chip BK3710 IDE Register Fail\n");
return -ENODEV;
}
hwif->set_pio_mode = &palm_bk3710_set_pio_mode;
hwif->set_dma_mode = &palm_bk3710_set_dma_mode;
hwif->mmio = 1;
default_hwif_mmiops(hwif);
hwif->cbl = ATA_CBL_PATA80;
hwif->ultra_mask = 0x1f; /* Ultra DMA Mode 4 Max
(input clk 99MHz) */
hwif->mwdma_mask = 0x7;
hwif->drives[0].autotune = 1;
hwif->drives[1].autotune = 1;
ide_setup_dma(hwif, mem->start);
return 0;
}
static struct platform_driver platform_bk_driver = {
.driver = {
.name = "palm_bk3710",
},
.probe = palm_bk3710_probe,
.remove = NULL,
};
static int __init palm_bk3710_init(void)
{
return platform_driver_register(&platform_bk_driver);
}
module_init(palm_bk3710_init);
MODULE_LICENSE("GPL");
...@@ -753,6 +753,25 @@ static void cris_set_dma_mode(ide_drive_t *drive, const u8 speed) ...@@ -753,6 +753,25 @@ static void cris_set_dma_mode(ide_drive_t *drive, const u8 speed)
cris_ide_set_speed(TYPE_DMA, 0, strobe, hold); cris_ide_set_speed(TYPE_DMA, 0, strobe, hold);
} }
static void __init cris_setup_ports(hw_regs_t *hw, unsigned long base)
{
int i;
memset(hw, 0, sizeof(*hw));
for (i = 0; i <= 7; i++)
hw->io_ports[i] = base + cris_ide_reg_addr(i, 0, 1);
/*
* the IDE control register is at ATA address 6,
* with CS1 active instead of CS0
*/
hw->io_ports[IDE_CONTROL_OFFSET] = base + cris_ide_reg_addr(6, 1, 0);
hw->irq = ide_default_irq(0);
hw->ack_intr = cris_ide_ack_intr;
}
static const struct ide_port_info cris_port_info __initdata = { static const struct ide_port_info cris_port_info __initdata = {
.chipset = ide_etrax100, .chipset = ide_etrax100,
.host_flags = IDE_HFLAG_NO_ATAPI_DMA | .host_flags = IDE_HFLAG_NO_ATAPI_DMA |
...@@ -765,24 +784,16 @@ static const struct ide_port_info cris_port_info __initdata = { ...@@ -765,24 +784,16 @@ static const struct ide_port_info cris_port_info __initdata = {
static int __init init_e100_ide(void) static int __init init_e100_ide(void)
{ {
hw_regs_t hw; hw_regs_t hw;
int ide_offsets[IDE_NR_PORTS], h, i; int h;
u8 idx[4] = { 0xff, 0xff, 0xff, 0xff }; u8 idx[4] = { 0xff, 0xff, 0xff, 0xff };
printk("ide: ETRAX FS built-in ATA DMA controller\n"); printk("ide: ETRAX FS built-in ATA DMA controller\n");
for (i = IDE_DATA_OFFSET; i <= IDE_STATUS_OFFSET; i++)
ide_offsets[i] = cris_ide_reg_addr(i, 0, 1);
/* the IDE control register is at ATA address 6, with CS1 active instead of CS0 */
ide_offsets[IDE_CONTROL_OFFSET] = cris_ide_reg_addr(6, 1, 0);
for (h = 0; h < 4; h++) { for (h = 0; h < 4; h++) {
ide_hwif_t *hwif = NULL; ide_hwif_t *hwif = NULL;
ide_setup_ports(&hw, cris_ide_base_address(h), cris_setup_ports(&hw, cris_ide_base_address(h));
ide_offsets,
0, 0, cris_ide_ack_intr,
ide_default_irq(0));
hwif = ide_find_port(hw.io_ports[IDE_DATA_OFFSET]); hwif = ide_find_port(hw.io_ports[IDE_DATA_OFFSET]);
if (hwif == NULL) if (hwif == NULL)
continue; continue;
......
...@@ -171,7 +171,7 @@ static int ide_get_dev_handle(struct device *dev, acpi_handle *handle, ...@@ -171,7 +171,7 @@ static int ide_get_dev_handle(struct device *dev, acpi_handle *handle,
static acpi_handle ide_acpi_hwif_get_handle(ide_hwif_t *hwif) static acpi_handle ide_acpi_hwif_get_handle(ide_hwif_t *hwif)
{ {
struct device *dev = hwif->gendev.parent; struct device *dev = hwif->gendev.parent;
acpi_handle dev_handle; acpi_handle uninitialized_var(dev_handle);
acpi_integer pcidevfn; acpi_integer pcidevfn;
acpi_handle chan_handle; acpi_handle chan_handle;
int err; int err;
......
...@@ -295,7 +295,8 @@ static int cdrom_decode_status(ide_drive_t *drive, int good_stat, int *stat_ret) ...@@ -295,7 +295,8 @@ static int cdrom_decode_status(ide_drive_t *drive, int good_stat, int *stat_ret)
int stat, err, sense_key; int stat, err, sense_key;
/* Check for errors. */ /* Check for errors. */
stat = HWIF(drive)->INB(IDE_STATUS_REG); stat = ide_read_status(drive);
if (stat_ret) if (stat_ret)
*stat_ret = stat; *stat_ret = stat;
...@@ -303,7 +304,7 @@ static int cdrom_decode_status(ide_drive_t *drive, int good_stat, int *stat_ret) ...@@ -303,7 +304,7 @@ static int cdrom_decode_status(ide_drive_t *drive, int good_stat, int *stat_ret)
return 0; return 0;
/* Get the IDE error register. */ /* Get the IDE error register. */
err = HWIF(drive)->INB(IDE_ERROR_REG); err = ide_read_error(drive);
sense_key = err >> 4; sense_key = err >> 4;
if (rq == NULL) { if (rq == NULL) {
...@@ -692,7 +693,7 @@ int ide_cd_check_ireason(ide_drive_t *drive, int len, int ireason, int rw) ...@@ -692,7 +693,7 @@ int ide_cd_check_ireason(ide_drive_t *drive, int len, int ireason, int rw)
/* Some drives (ASUS) seem to tell us that status /* Some drives (ASUS) seem to tell us that status
* info is available. just get it and ignore. * info is available. just get it and ignore.
*/ */
(void) HWIF(drive)->INB(IDE_STATUS_REG); (void)ide_read_status(drive);
return 0; return 0;
} else { } else {
/* Drive wants a command packet, or invalid ireason... */ /* Drive wants a command packet, or invalid ireason... */
...@@ -1326,7 +1327,7 @@ ide_do_rw_cdrom (ide_drive_t *drive, struct request *rq, sector_t block) ...@@ -1326,7 +1327,7 @@ ide_do_rw_cdrom (ide_drive_t *drive, struct request *rq, sector_t block)
if (blk_fs_request(rq)) { if (blk_fs_request(rq)) {
if (info->cd_flags & IDE_CD_FLAG_SEEKING) { if (info->cd_flags & IDE_CD_FLAG_SEEKING) {
unsigned long elapsed = jiffies - info->start_seek; unsigned long elapsed = jiffies - info->start_seek;
int stat = HWIF(drive)->INB(IDE_STATUS_REG); int stat = ide_read_status(drive);
if ((stat & SEEK_STAT) != SEEK_STAT) { if ((stat & SEEK_STAT) != SEEK_STAT) {
if (elapsed < IDECD_SEEK_TIMEOUT) { if (elapsed < IDECD_SEEK_TIMEOUT) {
......
...@@ -147,7 +147,8 @@ ide_startstop_t ide_dma_intr (ide_drive_t *drive) ...@@ -147,7 +147,8 @@ ide_startstop_t ide_dma_intr (ide_drive_t *drive)
u8 stat = 0, dma_stat = 0; u8 stat = 0, dma_stat = 0;
dma_stat = HWIF(drive)->ide_dma_end(drive); dma_stat = HWIF(drive)->ide_dma_end(drive);
stat = HWIF(drive)->INB(IDE_STATUS_REG); /* get drive status */ stat = ide_read_status(drive);
if (OK_STAT(stat,DRIVE_READY,drive->bad_wstat|DRQ_STAT)) { if (OK_STAT(stat,DRIVE_READY,drive->bad_wstat|DRQ_STAT)) {
if (!dma_stat) { if (!dma_stat) {
struct request *rq = HWGROUP(drive)->rq; struct request *rq = HWGROUP(drive)->rq;
......
...@@ -465,7 +465,7 @@ static void idefloppy_retry_pc(ide_drive_t *drive) ...@@ -465,7 +465,7 @@ static void idefloppy_retry_pc(ide_drive_t *drive)
idefloppy_pc_t *pc; idefloppy_pc_t *pc;
struct request *rq; struct request *rq;
(void)drive->hwif->INB(IDE_ERROR_REG); (void)ide_read_error(drive);
pc = idefloppy_next_pc_storage(drive); pc = idefloppy_next_pc_storage(drive);
rq = idefloppy_next_rq_storage(drive); rq = idefloppy_next_rq_storage(drive);
idefloppy_create_request_sense_cmd(pc); idefloppy_create_request_sense_cmd(pc);
...@@ -501,7 +501,7 @@ static ide_startstop_t idefloppy_pc_intr (ide_drive_t *drive) ...@@ -501,7 +501,7 @@ static ide_startstop_t idefloppy_pc_intr (ide_drive_t *drive)
} }
/* Clear the interrupt */ /* Clear the interrupt */
stat = drive->hwif->INB(IDE_STATUS_REG); stat = ide_read_status(drive);
/* No more interrupts */ /* No more interrupts */
if ((stat & DRQ_STAT) == 0) { if ((stat & DRQ_STAT) == 0) {
...@@ -1246,7 +1246,7 @@ static int idefloppy_get_format_progress(ide_drive_t *drive, int __user *arg) ...@@ -1246,7 +1246,7 @@ static int idefloppy_get_format_progress(ide_drive_t *drive, int __user *arg)
u8 stat; u8 stat;
local_irq_save(flags); local_irq_save(flags);
stat = drive->hwif->INB(IDE_STATUS_REG); stat = ide_read_status(drive);
local_irq_restore(flags); local_irq_restore(flags);
progress_indication = ((stat & SEEK_STAT) == 0) ? 0 : 0x10000; progress_indication = ((stat & SEEK_STAT) == 0) ? 0 : 0x10000;
......
...@@ -20,8 +20,14 @@ static int __init ide_generic_init(void) ...@@ -20,8 +20,14 @@ static int __init ide_generic_init(void)
if (ide_hwifs[0].io_ports[IDE_DATA_OFFSET]) if (ide_hwifs[0].io_ports[IDE_DATA_OFFSET])
ide_get_lock(NULL, NULL); /* for atari only */ ide_get_lock(NULL, NULL); /* for atari only */
for (i = 0; i < MAX_HWIFS; i++) for (i = 0; i < MAX_HWIFS; i++) {
idx[i] = ide_hwifs[i].present ? 0xff : i; ide_hwif_t *hwif = &ide_hwifs[i];
if (hwif->io_ports[IDE_DATA_OFFSET] && !hwif->present)
idx[i] = i;
else
idx[i] = 0xff;
}
ide_device_add_all(idx, NULL); ide_device_add_all(idx, NULL);
......
...@@ -466,7 +466,7 @@ static ide_startstop_t ide_ata_error(ide_drive_t *drive, struct request *rq, u8 ...@@ -466,7 +466,7 @@ static ide_startstop_t ide_ata_error(ide_drive_t *drive, struct request *rq, u8
return ide_stopped; return ide_stopped;
} }
if (hwif->INB(IDE_STATUS_REG) & (BUSY_STAT|DRQ_STAT)) if (ide_read_status(drive) & (BUSY_STAT | DRQ_STAT))
rq->errors |= ERROR_RESET; rq->errors |= ERROR_RESET;
if ((rq->errors & ERROR_RESET) == ERROR_RESET) { if ((rq->errors & ERROR_RESET) == ERROR_RESET) {
...@@ -493,7 +493,7 @@ static ide_startstop_t ide_atapi_error(ide_drive_t *drive, struct request *rq, u ...@@ -493,7 +493,7 @@ static ide_startstop_t ide_atapi_error(ide_drive_t *drive, struct request *rq, u
/* add decoding error stuff */ /* add decoding error stuff */
} }
if (hwif->INB(IDE_STATUS_REG) & (BUSY_STAT|DRQ_STAT)) if (ide_read_status(drive) & (BUSY_STAT | DRQ_STAT))
/* force an abort */ /* force an abort */
hwif->OUTB(WIN_IDLEIMMEDIATE, IDE_COMMAND_REG); hwif->OUTB(WIN_IDLEIMMEDIATE, IDE_COMMAND_REG);
...@@ -821,9 +821,8 @@ static ide_startstop_t execute_drive_cmd (ide_drive_t *drive, ...@@ -821,9 +821,8 @@ static ide_startstop_t execute_drive_cmd (ide_drive_t *drive,
#ifdef DEBUG #ifdef DEBUG
printk("%s: DRIVE_CMD (null)\n", drive->name); printk("%s: DRIVE_CMD (null)\n", drive->name);
#endif #endif
ide_end_drive_cmd(drive, ide_end_drive_cmd(drive, ide_read_status(drive), ide_read_error(drive));
hwif->INB(IDE_STATUS_REG),
hwif->INB(IDE_ERROR_REG));
return ide_stopped; return ide_stopped;
} }
...@@ -1231,7 +1230,7 @@ static ide_startstop_t ide_dma_timeout_retry(ide_drive_t *drive, int error) ...@@ -1231,7 +1230,7 @@ static ide_startstop_t ide_dma_timeout_retry(ide_drive_t *drive, int error)
printk(KERN_WARNING "%s: DMA timeout error\n", drive->name); printk(KERN_WARNING "%s: DMA timeout error\n", drive->name);
(void)HWIF(drive)->ide_dma_end(drive); (void)HWIF(drive)->ide_dma_end(drive);
ret = ide_error(drive, "dma timeout error", ret = ide_error(drive, "dma timeout error",
hwif->INB(IDE_STATUS_REG)); ide_read_status(drive));
} else { } else {
printk(KERN_WARNING "%s: DMA timeout retry\n", drive->name); printk(KERN_WARNING "%s: DMA timeout retry\n", drive->name);
hwif->dma_timeout(drive); hwif->dma_timeout(drive);
...@@ -1355,7 +1354,8 @@ void ide_timer_expiry (unsigned long data) ...@@ -1355,7 +1354,8 @@ void ide_timer_expiry (unsigned long data)
startstop = ide_dma_timeout_retry(drive, wait); startstop = ide_dma_timeout_retry(drive, wait);
} else } else
startstop = startstop =
ide_error(drive, "irq timeout", hwif->INB(IDE_STATUS_REG)); ide_error(drive, "irq timeout",
ide_read_status(drive));
} }
drive->service_time = jiffies - drive->service_start; drive->service_time = jiffies - drive->service_start;
spin_lock_irq(&ide_lock); spin_lock_irq(&ide_lock);
......
...@@ -430,10 +430,10 @@ int drive_is_ready (ide_drive_t *drive) ...@@ -430,10 +430,10 @@ int drive_is_ready (ide_drive_t *drive)
* about possible isa-pnp and pci-pnp issues yet. * about possible isa-pnp and pci-pnp issues yet.
*/ */
if (IDE_CONTROL_REG) if (IDE_CONTROL_REG)
stat = hwif->INB(IDE_ALTSTATUS_REG); stat = ide_read_altstatus(drive);
else else
/* Note: this may clear a pending IRQ!! */ /* Note: this may clear a pending IRQ!! */
stat = hwif->INB(IDE_STATUS_REG); stat = ide_read_status(drive);
if (stat & BUSY_STAT) if (stat & BUSY_STAT)
/* drive busy: definitely not interrupting */ /* drive busy: definitely not interrupting */
...@@ -458,23 +458,24 @@ EXPORT_SYMBOL(drive_is_ready); ...@@ -458,23 +458,24 @@ EXPORT_SYMBOL(drive_is_ready);
*/ */
static int __ide_wait_stat(ide_drive_t *drive, u8 good, u8 bad, unsigned long timeout, u8 *rstat) static int __ide_wait_stat(ide_drive_t *drive, u8 good, u8 bad, unsigned long timeout, u8 *rstat)
{ {
ide_hwif_t *hwif = drive->hwif;
unsigned long flags; unsigned long flags;
int i; int i;
u8 stat; u8 stat;
udelay(1); /* spec allows drive 400ns to assert "BUSY" */ udelay(1); /* spec allows drive 400ns to assert "BUSY" */
if ((stat = hwif->INB(IDE_STATUS_REG)) & BUSY_STAT) { stat = ide_read_status(drive);
if (stat & BUSY_STAT) {
local_irq_set(flags); local_irq_set(flags);
timeout += jiffies; timeout += jiffies;
while ((stat = hwif->INB(IDE_STATUS_REG)) & BUSY_STAT) { while ((stat = ide_read_status(drive)) & BUSY_STAT) {
if (time_after(jiffies, timeout)) { if (time_after(jiffies, timeout)) {
/* /*
* One last read after the timeout in case * One last read after the timeout in case
* heavy interrupt load made us not make any * heavy interrupt load made us not make any
* progress during the timeout.. * progress during the timeout..
*/ */
stat = hwif->INB(IDE_STATUS_REG); stat = ide_read_status(drive);
if (!(stat & BUSY_STAT)) if (!(stat & BUSY_STAT))
break; break;
...@@ -494,7 +495,9 @@ static int __ide_wait_stat(ide_drive_t *drive, u8 good, u8 bad, unsigned long ti ...@@ -494,7 +495,9 @@ static int __ide_wait_stat(ide_drive_t *drive, u8 good, u8 bad, unsigned long ti
*/ */
for (i = 0; i < 10; i++) { for (i = 0; i < 10; i++) {
udelay(1); udelay(1);
if (OK_STAT((stat = hwif->INB(IDE_STATUS_REG)), good, bad)) { stat = ide_read_status(drive);
if (OK_STAT(stat, good, bad)) {
*rstat = stat; *rstat = stat;
return 0; return 0;
} }
...@@ -617,6 +620,7 @@ int ide_driveid_update(ide_drive_t *drive) ...@@ -617,6 +620,7 @@ int ide_driveid_update(ide_drive_t *drive)
ide_hwif_t *hwif = drive->hwif; ide_hwif_t *hwif = drive->hwif;
struct hd_driveid *id; struct hd_driveid *id;
unsigned long timeout, flags; unsigned long timeout, flags;
u8 stat;
/* /*
* Re-read drive->id for possible DMA mode * Re-read drive->id for possible DMA mode
...@@ -633,10 +637,15 @@ int ide_driveid_update(ide_drive_t *drive) ...@@ -633,10 +637,15 @@ int ide_driveid_update(ide_drive_t *drive)
SELECT_MASK(drive, 0); SELECT_MASK(drive, 0);
return 0; /* drive timed-out */ return 0; /* drive timed-out */
} }
msleep(50); /* give drive a breather */ msleep(50); /* give drive a breather */
} while (hwif->INB(IDE_ALTSTATUS_REG) & BUSY_STAT); stat = ide_read_altstatus(drive);
} while (stat & BUSY_STAT);
msleep(50); /* wait for IRQ and DRQ_STAT */ msleep(50); /* wait for IRQ and DRQ_STAT */
if (!OK_STAT(hwif->INB(IDE_STATUS_REG),DRQ_STAT,BAD_R_STAT)) { stat = ide_read_status(drive);
if (!OK_STAT(stat, DRQ_STAT, BAD_R_STAT)) {
SELECT_MASK(drive, 0); SELECT_MASK(drive, 0);
printk("%s: CHECK for good STATUS\n", drive->name); printk("%s: CHECK for good STATUS\n", drive->name);
return 0; return 0;
...@@ -649,7 +658,7 @@ int ide_driveid_update(ide_drive_t *drive) ...@@ -649,7 +658,7 @@ int ide_driveid_update(ide_drive_t *drive)
return 0; return 0;
} }
ata_input_data(drive, id, SECTOR_WORDS); ata_input_data(drive, id, SECTOR_WORDS);
(void) hwif->INB(IDE_STATUS_REG); /* clear drive IRQ */ (void)ide_read_status(drive); /* clear drive IRQ */
local_irq_enable(); local_irq_enable();
local_irq_restore(flags); local_irq_restore(flags);
ide_fix_driveid(id); ide_fix_driveid(id);
...@@ -850,17 +859,16 @@ static ide_startstop_t do_reset1 (ide_drive_t *, int); ...@@ -850,17 +859,16 @@ static ide_startstop_t do_reset1 (ide_drive_t *, int);
static ide_startstop_t atapi_reset_pollfunc (ide_drive_t *drive) static ide_startstop_t atapi_reset_pollfunc (ide_drive_t *drive)
{ {
ide_hwgroup_t *hwgroup = HWGROUP(drive); ide_hwgroup_t *hwgroup = HWGROUP(drive);
ide_hwif_t *hwif = HWIF(drive);
u8 stat; u8 stat;
SELECT_DRIVE(drive); SELECT_DRIVE(drive);
udelay (10); udelay (10);
stat = ide_read_status(drive);
if (OK_STAT(stat = hwif->INB(IDE_STATUS_REG), 0, BUSY_STAT)) { if (OK_STAT(stat, 0, BUSY_STAT))
printk("%s: ATAPI reset complete\n", drive->name); printk("%s: ATAPI reset complete\n", drive->name);
} else { else {
if (time_before(jiffies, hwgroup->poll_timeout)) { if (time_before(jiffies, hwgroup->poll_timeout)) {
BUG_ON(HWGROUP(drive)->handler != NULL);
ide_set_handler(drive, &atapi_reset_pollfunc, HZ/20, NULL); ide_set_handler(drive, &atapi_reset_pollfunc, HZ/20, NULL);
/* continue polling */ /* continue polling */
return ide_started; return ide_started;
...@@ -898,9 +906,10 @@ static ide_startstop_t reset_pollfunc (ide_drive_t *drive) ...@@ -898,9 +906,10 @@ static ide_startstop_t reset_pollfunc (ide_drive_t *drive)
} }
} }
if (!OK_STAT(tmp = hwif->INB(IDE_STATUS_REG), 0, BUSY_STAT)) { tmp = ide_read_status(drive);
if (!OK_STAT(tmp, 0, BUSY_STAT)) {
if (time_before(jiffies, hwgroup->poll_timeout)) { if (time_before(jiffies, hwgroup->poll_timeout)) {
BUG_ON(HWGROUP(drive)->handler != NULL);
ide_set_handler(drive, &reset_pollfunc, HZ/20, NULL); ide_set_handler(drive, &reset_pollfunc, HZ/20, NULL);
/* continue polling */ /* continue polling */
return ide_started; return ide_started;
...@@ -909,7 +918,9 @@ static ide_startstop_t reset_pollfunc (ide_drive_t *drive) ...@@ -909,7 +918,9 @@ static ide_startstop_t reset_pollfunc (ide_drive_t *drive)
drive->failures++; drive->failures++;
} else { } else {
printk("%s: reset: ", hwif->name); printk("%s: reset: ", hwif->name);
if ((tmp = hwif->INB(IDE_ERROR_REG)) == 1) { tmp = ide_read_error(drive);
if (tmp == 1) {
printk("success\n"); printk("success\n");
drive->failures = 0; drive->failures = 0;
} else { } else {
......
...@@ -578,7 +578,7 @@ u8 ide_dump_status(ide_drive_t *drive, const char *msg, u8 stat) ...@@ -578,7 +578,7 @@ u8 ide_dump_status(ide_drive_t *drive, const char *msg, u8 stat)
} }
printk("}\n"); printk("}\n");
if ((stat & (BUSY_STAT|ERR_STAT)) == ERR_STAT) { if ((stat & (BUSY_STAT|ERR_STAT)) == ERR_STAT) {
err = drive->hwif->INB(IDE_ERROR_REG); err = ide_read_error(drive);
printk("%s: %s: error=0x%02x ", drive->name, msg, err); printk("%s: %s: error=0x%02x ", drive->name, msg, err);
if (drive->media == ide_disk) if (drive->media == ide_disk)
ide_dump_ata_error(drive, err); ide_dump_ata_error(drive, err);
......
...@@ -264,8 +264,7 @@ static inline void do_identify (ide_drive_t *drive, u8 cmd) ...@@ -264,8 +264,7 @@ static inline void do_identify (ide_drive_t *drive, u8 cmd)
static int actual_try_to_identify (ide_drive_t *drive, u8 cmd) static int actual_try_to_identify (ide_drive_t *drive, u8 cmd)
{ {
ide_hwif_t *hwif = HWIF(drive); ide_hwif_t *hwif = HWIF(drive);
int rc; int use_altstatus = 0, rc;
unsigned long hd_status;
unsigned long timeout; unsigned long timeout;
u8 s = 0, a = 0; u8 s = 0, a = 0;
...@@ -273,19 +272,17 @@ static int actual_try_to_identify (ide_drive_t *drive, u8 cmd) ...@@ -273,19 +272,17 @@ static int actual_try_to_identify (ide_drive_t *drive, u8 cmd)
msleep(50); msleep(50);
if (IDE_CONTROL_REG) { if (IDE_CONTROL_REG) {
a = hwif->INB(IDE_ALTSTATUS_REG); a = ide_read_altstatus(drive);
s = hwif->INB(IDE_STATUS_REG); s = ide_read_status(drive);
if ((a ^ s) & ~INDEX_STAT) { if ((a ^ s) & ~INDEX_STAT)
printk(KERN_INFO "%s: probing with STATUS(0x%02x) instead of "
"ALTSTATUS(0x%02x)\n", drive->name, s, a);
/* ancient Seagate drives, broken interfaces */ /* ancient Seagate drives, broken interfaces */
hd_status = IDE_STATUS_REG; printk(KERN_INFO "%s: probing with STATUS(0x%02x) "
} else { "instead of ALTSTATUS(0x%02x)\n",
drive->name, s, a);
else
/* use non-intrusive polling */ /* use non-intrusive polling */
hd_status = IDE_ALTSTATUS_REG; use_altstatus = 1;
} }
} else
hd_status = IDE_STATUS_REG;
/* set features register for atapi /* set features register for atapi
* identify command to be sure of reply * identify command to be sure of reply
...@@ -306,11 +303,15 @@ static int actual_try_to_identify (ide_drive_t *drive, u8 cmd) ...@@ -306,11 +303,15 @@ static int actual_try_to_identify (ide_drive_t *drive, u8 cmd)
} }
/* give drive a breather */ /* give drive a breather */
msleep(50); msleep(50);
} while ((hwif->INB(hd_status)) & BUSY_STAT); s = use_altstatus ? ide_read_altstatus(drive)
: ide_read_status(drive);
} while (s & BUSY_STAT);
/* wait for IRQ and DRQ_STAT */ /* wait for IRQ and DRQ_STAT */
msleep(50); msleep(50);
if (OK_STAT((hwif->INB(IDE_STATUS_REG)), DRQ_STAT, BAD_R_STAT)) { s = ide_read_status(drive);
if (OK_STAT(s, DRQ_STAT, BAD_R_STAT)) {
unsigned long flags; unsigned long flags;
/* local CPU only; some systems need this */ /* local CPU only; some systems need this */
...@@ -320,7 +321,7 @@ static int actual_try_to_identify (ide_drive_t *drive, u8 cmd) ...@@ -320,7 +321,7 @@ static int actual_try_to_identify (ide_drive_t *drive, u8 cmd)
/* drive responded with ID */ /* drive responded with ID */
rc = 0; rc = 0;
/* clear drive IRQ */ /* clear drive IRQ */
(void) hwif->INB(IDE_STATUS_REG); (void)ide_read_status(drive);
local_irq_restore(flags); local_irq_restore(flags);
} else { } else {
/* drive refused ID */ /* drive refused ID */
...@@ -367,7 +368,7 @@ static int try_to_identify (ide_drive_t *drive, u8 cmd) ...@@ -367,7 +368,7 @@ static int try_to_identify (ide_drive_t *drive, u8 cmd)
ide_set_irq(drive, 0); ide_set_irq(drive, 0);
/* clear drive IRQ */ /* clear drive IRQ */
(void) hwif->INB(IDE_STATUS_REG); (void)ide_read_status(drive);
udelay(5); udelay(5);
irq = probe_irq_off(cookie); irq = probe_irq_off(cookie);
if (!hwif->irq) { if (!hwif->irq) {
...@@ -455,7 +456,9 @@ static int do_probe (ide_drive_t *drive, u8 cmd) ...@@ -455,7 +456,9 @@ static int do_probe (ide_drive_t *drive, u8 cmd)
return 3; return 3;
} }
if (OK_STAT((hwif->INB(IDE_STATUS_REG)), READY_STAT, BUSY_STAT) || stat = ide_read_status(drive);
if (OK_STAT(stat, READY_STAT, BUSY_STAT) ||
drive->present || cmd == WIN_PIDENTIFY) { drive->present || cmd == WIN_PIDENTIFY) {
/* send cmd and wait */ /* send cmd and wait */
if ((rc = try_to_identify(drive, cmd))) { if ((rc = try_to_identify(drive, cmd))) {
...@@ -463,7 +466,7 @@ static int do_probe (ide_drive_t *drive, u8 cmd) ...@@ -463,7 +466,7 @@ static int do_probe (ide_drive_t *drive, u8 cmd)
rc = try_to_identify(drive,cmd); rc = try_to_identify(drive,cmd);
} }
stat = hwif->INB(IDE_STATUS_REG); stat = ide_read_status(drive);
if (stat == (BUSY_STAT | READY_STAT)) if (stat == (BUSY_STAT | READY_STAT))
return 4; return 4;
...@@ -482,7 +485,7 @@ static int do_probe (ide_drive_t *drive, u8 cmd) ...@@ -482,7 +485,7 @@ static int do_probe (ide_drive_t *drive, u8 cmd)
} }
/* ensure drive IRQ is clear */ /* ensure drive IRQ is clear */
stat = hwif->INB(IDE_STATUS_REG); stat = ide_read_status(drive);
if (rc == 1) if (rc == 1)
printk(KERN_ERR "%s: no response (status = 0x%02x)\n", printk(KERN_ERR "%s: no response (status = 0x%02x)\n",
...@@ -496,7 +499,7 @@ static int do_probe (ide_drive_t *drive, u8 cmd) ...@@ -496,7 +499,7 @@ static int do_probe (ide_drive_t *drive, u8 cmd)
SELECT_DRIVE(&hwif->drives[0]); SELECT_DRIVE(&hwif->drives[0]);
msleep(50); msleep(50);
/* ensure drive irq is clear */ /* ensure drive irq is clear */
(void) hwif->INB(IDE_STATUS_REG); (void)ide_read_status(drive);
} }
return rc; return rc;
} }
...@@ -521,7 +524,7 @@ static void enable_nest (ide_drive_t *drive) ...@@ -521,7 +524,7 @@ static void enable_nest (ide_drive_t *drive)
msleep(50); msleep(50);
stat = hwif->INB(IDE_STATUS_REG); stat = ide_read_status(drive);
if (!OK_STAT(stat, 0, BAD_STAT)) if (!OK_STAT(stat, 0, BAD_STAT))
printk(KERN_CONT "failed (status = 0x%02x)\n", stat); printk(KERN_CONT "failed (status = 0x%02x)\n", stat);
......
...@@ -65,6 +65,7 @@ static int proc_ide_read_imodel ...@@ -65,6 +65,7 @@ static int proc_ide_read_imodel
case ide_4drives: name = "4drives"; break; case ide_4drives: name = "4drives"; break;
case ide_pmac: name = "mac-io"; break; case ide_pmac: name = "mac-io"; break;
case ide_au1xxx: name = "au1xxx"; break; case ide_au1xxx: name = "au1xxx"; break;
case ide_palm3710: name = "palm3710"; break;
case ide_etrax100: name = "etrax100"; break; case ide_etrax100: name = "etrax100"; break;
case ide_acorn: name = "acorn"; break; case ide_acorn: name = "acorn"; break;
default: name = "(unknown)"; break; default: name = "(unknown)"; break;
......
...@@ -15,7 +15,7 @@ ...@@ -15,7 +15,7 @@
* Documentation/ide/ChangeLog.ide-tape.1995-2002 * Documentation/ide/ChangeLog.ide-tape.1995-2002
*/ */
#define IDETAPE_VERSION "1.19" #define IDETAPE_VERSION "1.20"
#include <linux/module.h> #include <linux/module.h>
#include <linux/types.h> #include <linux/types.h>
...@@ -39,63 +39,70 @@ ...@@ -39,63 +39,70 @@
#include <scsi/scsi.h> #include <scsi/scsi.h>
#include <asm/byteorder.h> #include <asm/byteorder.h>
#include <asm/irq.h> #include <linux/irq.h>
#include <asm/uaccess.h> #include <linux/uaccess.h>
#include <asm/io.h> #include <linux/io.h>
#include <asm/unaligned.h> #include <asm/unaligned.h>
#include <linux/mtio.h> #include <linux/mtio.h>
enum {
/* output errors only */
DBG_ERR = (1 << 0),
/* output all sense key/asc */
DBG_SENSE = (1 << 1),
/* info regarding all chrdev-related procedures */
DBG_CHRDEV = (1 << 2),
/* all remaining procedures */
DBG_PROCS = (1 << 3),
/* buffer alloc info (pc_stack & rq_stack) */
DBG_PCRQ_STACK = (1 << 4),
};
/* define to see debug info */
#define IDETAPE_DEBUG_LOG 0
#if IDETAPE_DEBUG_LOG
#define debug_log(lvl, fmt, args...) \
{ \
if (tape->debug_mask & lvl) \
printk(KERN_INFO "ide-tape: " fmt, ## args); \
}
#else
#define debug_log(lvl, fmt, args...) do {} while (0)
#endif
/**************************** Tunable parameters *****************************/ /**************************** Tunable parameters *****************************/
/* /*
* Pipelined mode parameters. * Pipelined mode parameters.
* *
* We try to use the minimum number of stages which is enough to * We try to use the minimum number of stages which is enough to keep the tape
* keep the tape constantly streaming. To accomplish that, we implement * constantly streaming. To accomplish that, we implement a feedback loop around
* a feedback loop around the maximum number of stages: * the maximum number of stages:
* *
* We start from MIN maximum stages (we will not even use MIN stages * We start from MIN maximum stages (we will not even use MIN stages if we don't
* if we don't need them), increment it by RATE*(MAX-MIN) * need them), increment it by RATE*(MAX-MIN) whenever we sense that the
* whenever we sense that the pipeline is empty, until we reach * pipeline is empty, until we reach the optimum value or until we reach MAX.
* the optimum value or until we reach MAX.
* *
* Setting the following parameter to 0 is illegal: the pipelined mode * Setting the following parameter to 0 is illegal: the pipelined mode cannot be
* cannot be disabled (calculate_speeds() divides by tape->max_stages.) * disabled (idetape_calculate_speeds() divides by tape->max_stages.)
*/ */
#define IDETAPE_MIN_PIPELINE_STAGES 1 #define IDETAPE_MIN_PIPELINE_STAGES 1
#define IDETAPE_MAX_PIPELINE_STAGES 400 #define IDETAPE_MAX_PIPELINE_STAGES 400
#define IDETAPE_INCREASE_STAGES_RATE 20 #define IDETAPE_INCREASE_STAGES_RATE 20
/* /*
* The following are used to debug the driver: * After each failed packet command we issue a request sense command and retry
* * the packet command IDETAPE_MAX_PC_RETRIES times.
* Setting IDETAPE_DEBUG_LOG to 1 will log driver flow control.
* *
* Setting them to 0 will restore normal operation mode: * Setting IDETAPE_MAX_PC_RETRIES to 0 will disable retries.
*
* 1. Disable logging normal successful operations.
* 2. Disable self-sanity checks.
* 3. Errors will still be logged, of course.
*
* All the #if DEBUG code will be removed some day, when the driver
* is verified to be stable enough. This will make it much more
* esthetic.
*/
#define IDETAPE_DEBUG_LOG 0
/*
* After each failed packet command we issue a request sense command
* and retry the packet command IDETAPE_MAX_PC_RETRIES times.
*
* Setting IDETAPE_MAX_PC_RETRIES to 0 will disable retries.
*/ */
#define IDETAPE_MAX_PC_RETRIES 3 #define IDETAPE_MAX_PC_RETRIES 3
/* /*
* With each packet command, we allocate a buffer of * With each packet command, we allocate a buffer of IDETAPE_PC_BUFFER_SIZE
* IDETAPE_PC_BUFFER_SIZE bytes. This is used for several packet * bytes. This is used for several packet commands (Not for READ/WRITE commands)
* commands (Not for READ/WRITE commands).
*/ */
#define IDETAPE_PC_BUFFER_SIZE 256 #define IDETAPE_PC_BUFFER_SIZE 256
...@@ -114,48 +121,39 @@ ...@@ -114,48 +121,39 @@
#define IDETAPE_WAIT_CMD (900*HZ) #define IDETAPE_WAIT_CMD (900*HZ)
/* /*
* The following parameter is used to select the point in the internal * The following parameter is used to select the point in the internal tape fifo
* tape fifo in which we will start to refill the buffer. Decreasing * in which we will start to refill the buffer. Decreasing the following
* the following parameter will improve the system's latency and * parameter will improve the system's latency and interactive response, while
* interactive response, while using a high value might improve system * using a high value might improve system throughput.
* throughput.
*/ */
#define IDETAPE_FIFO_THRESHOLD 2 #define IDETAPE_FIFO_THRESHOLD 2
/* /*
* DSC polling parameters. * DSC polling parameters.
*
* Polling for DSC (a single bit in the status register) is a very
* important function in ide-tape. There are two cases in which we
* poll for DSC:
* *
* 1. Before a read/write packet command, to ensure that we * Polling for DSC (a single bit in the status register) is a very important
* can transfer data from/to the tape's data buffers, without * function in ide-tape. There are two cases in which we poll for DSC:
* causing an actual media access. In case the tape is not
* ready yet, we take out our request from the device
* request queue, so that ide.c will service requests from
* the other device on the same interface meanwhile.
* *
* 2. After the successful initialization of a "media access * 1. Before a read/write packet command, to ensure that we can transfer data
* packet command", which is a command which can take a long * from/to the tape's data buffers, without causing an actual media access.
* time to complete (it can be several seconds or even an hour). * In case the tape is not ready yet, we take out our request from the device
* request queue, so that ide.c could service requests from the other device
* on the same interface in the meantime.
* *
* Again, we postpone our request in the middle to free the bus * 2. After the successful initialization of a "media access packet command",
* for the other device. The polling frequency here should be * which is a command that can take a long time to complete (the interval can
* lower than the read/write frequency since those media access * range from several seconds to even an hour). Again, we postpone our request
* commands are slow. We start from a "fast" frequency - * in the middle to free the bus for the other device. The polling frequency
* IDETAPE_DSC_MA_FAST (one second), and if we don't receive DSC * here should be lower than the read/write frequency since those media access
* after IDETAPE_DSC_MA_THRESHOLD (5 minutes), we switch it to a * commands are slow. We start from a "fast" frequency - IDETAPE_DSC_MA_FAST
* lower frequency - IDETAPE_DSC_MA_SLOW (1 minute). * (1 second), and if we don't receive DSC after IDETAPE_DSC_MA_THRESHOLD
* (5 min), we switch it to a lower frequency - IDETAPE_DSC_MA_SLOW (1 min).
* *
* We also set a timeout for the timer, in case something goes wrong. * We also set a timeout for the timer, in case something goes wrong. The
* The timeout should be longer then the maximum execution time of a * timeout should be longer then the maximum execution time of a tape operation.
* tape operation.
*/
/*
* DSC timings.
*/ */
/* DSC timings. */
#define IDETAPE_DSC_RW_MIN 5*HZ/100 /* 50 msec */ #define IDETAPE_DSC_RW_MIN 5*HZ/100 /* 50 msec */
#define IDETAPE_DSC_RW_MAX 40*HZ/100 /* 400 msec */ #define IDETAPE_DSC_RW_MAX 40*HZ/100 /* 400 msec */
#define IDETAPE_DSC_RW_TIMEOUT 2*60*HZ /* 2 minutes */ #define IDETAPE_DSC_RW_TIMEOUT 2*60*HZ /* 2 minutes */
...@@ -166,19 +164,15 @@ ...@@ -166,19 +164,15 @@
/*************************** End of tunable parameters ***********************/ /*************************** End of tunable parameters ***********************/
/* /* Read/Write error simulation */
* Read/Write error simulation
*/
#define SIMULATE_ERRORS 0 #define SIMULATE_ERRORS 0
/* /* tape directions */
* For general magnetic tape device compatibility. enum {
*/ IDETAPE_DIR_NONE = (1 << 0),
typedef enum { IDETAPE_DIR_READ = (1 << 1),
idetape_direction_none, IDETAPE_DIR_WRITE = (1 << 2),
idetape_direction_read, };
idetape_direction_write
} idetape_chrdev_direction_t;
struct idetape_bh { struct idetape_bh {
u32 b_size; u32 b_size;
...@@ -187,24 +181,32 @@ struct idetape_bh { ...@@ -187,24 +181,32 @@ struct idetape_bh {
char *b_data; char *b_data;
}; };
/*
* Our view of a packet command.
*/
typedef struct idetape_packet_command_s { typedef struct idetape_packet_command_s {
u8 c[12]; /* Actual packet bytes */ /* Actual packet bytes */
int retries; /* On each retry, we increment retries */ u8 c[12];
int error; /* Error code */ /* On each retry, we increment retries */
int request_transfer; /* Bytes to transfer */ int retries;
int actually_transferred; /* Bytes actually transferred */ /* Error code */
int buffer_size; /* Size of our data buffer */ int error;
/* Bytes to transfer */
int request_transfer;
/* Bytes actually transferred */
int actually_transferred;
/* Size of our data buffer */
int buffer_size;
struct idetape_bh *bh; struct idetape_bh *bh;
char *b_data; char *b_data;
int b_count; int b_count;
u8 *buffer; /* Data buffer */ /* Data buffer */
u8 *current_position; /* Pointer into the above buffer */ u8 *buffer;
ide_startstop_t (*callback) (ide_drive_t *); /* Called when this packet command is completed */ /* Pointer into the above buffer */
u8 pc_buffer[IDETAPE_PC_BUFFER_SIZE]; /* Temporary buffer */ u8 *current_position;
unsigned long flags; /* Status/Action bit flags: long for set_bit */ /* Called when this packet command is completed */
ide_startstop_t (*callback) (ide_drive_t *);
/* Temporary buffer */
u8 pc_buffer[IDETAPE_PC_BUFFER_SIZE];
/* Status/Action bit flags: long for set_bit */
unsigned long flags;
} idetape_pc_t; } idetape_pc_t;
/* /*
...@@ -223,9 +225,7 @@ typedef struct idetape_packet_command_s { ...@@ -223,9 +225,7 @@ typedef struct idetape_packet_command_s {
/* Data direction */ /* Data direction */
#define PC_WRITING 5 #define PC_WRITING 5
/* /* A pipeline stage. */
* A pipeline stage.
*/
typedef struct idetape_stage_s { typedef struct idetape_stage_s {
struct request rq; /* The corresponding request */ struct request rq; /* The corresponding request */
struct idetape_bh *bh; /* The data buffers */ struct idetape_bh *bh; /* The data buffers */
...@@ -233,9 +233,8 @@ typedef struct idetape_stage_s { ...@@ -233,9 +233,8 @@ typedef struct idetape_stage_s {
} idetape_stage_t; } idetape_stage_t;
/* /*
* Most of our global data which we need to save even as we leave the * Most of our global data which we need to save even as we leave the driver due
* driver due to an interrupt or a timer event is stored in a variable * to an interrupt or a timer event is stored in the struct defined below.
* of type idetape_tape_t, defined below.
*/ */
typedef struct ide_tape_obj { typedef struct ide_tape_obj {
ide_drive_t *drive; ide_drive_t *drive;
...@@ -271,15 +270,14 @@ typedef struct ide_tape_obj { ...@@ -271,15 +270,14 @@ typedef struct ide_tape_obj {
int rq_stack_index; int rq_stack_index;
/* /*
* DSC polling variables. * DSC polling variables.
* *
* While polling for DSC we use postponed_rq to postpone the * While polling for DSC we use postponed_rq to postpone the current
* current request so that ide.c will be able to service * request so that ide.c will be able to service pending requests on the
* pending requests on the other device. Note that at most * other device. Note that at most we will have only one DSC (usually
* we will have only one DSC (usually data transfer) request * data transfer) request in the device request queue. Additional
* in the device request queue. Additional requests can be * requests can be queued in our internal pipeline, but they will be
* queued in our internal pipeline, but they will be visible * visible to ide.c only one at a time.
* to ide.c only one at a time.
*/ */
struct request *postponed_rq; struct request *postponed_rq;
/* The time in which we started polling for DSC */ /* The time in which we started polling for DSC */
...@@ -287,73 +285,57 @@ typedef struct ide_tape_obj { ...@@ -287,73 +285,57 @@ typedef struct ide_tape_obj {
/* Timer used to poll for dsc */ /* Timer used to poll for dsc */
struct timer_list dsc_timer; struct timer_list dsc_timer;
/* Read/Write dsc polling frequency */ /* Read/Write dsc polling frequency */
unsigned long best_dsc_rw_frequency; unsigned long best_dsc_rw_freq;
/* The current polling frequency */ unsigned long dsc_poll_freq;
unsigned long dsc_polling_frequency;
/* Maximum waiting time */
unsigned long dsc_timeout; unsigned long dsc_timeout;
/* /* Read position information */
* Read position information
*/
u8 partition; u8 partition;
/* Current block */ /* Current block */
unsigned int first_frame_position; unsigned int first_frame;
unsigned int last_frame_position;
unsigned int blocks_in_buffer;
/* /* Last error information */
* Last error information
*/
u8 sense_key, asc, ascq; u8 sense_key, asc, ascq;
/* /* Character device operation */
* Character device operation
*/
unsigned int minor; unsigned int minor;
/* device name */ /* device name */
char name[4]; char name[4];
/* Current character device data transfer direction */ /* Current character device data transfer direction */
idetape_chrdev_direction_t chrdev_direction; u8 chrdev_dir;
/* /* tape block size, usually 512 or 1024 bytes */
* Device information unsigned short blk_size;
*/
/* Usually 512 or 1024 bytes */
unsigned short tape_block_size;
int user_bs_factor; int user_bs_factor;
/* Copy of the tape's Capabilities and Mechanical Page */ /* Copy of the tape's Capabilities and Mechanical Page */
u8 caps[20]; u8 caps[20];
/* /*
* Active data transfer request parameters. * Active data transfer request parameters.
*
* At most, there is only one ide-tape originated data transfer
* request in the device request queue. This allows ide.c to
* easily service requests from the other device when we
* postpone our active request. In the pipelined operation
* mode, we use our internal pipeline structure to hold
* more data requests.
* *
* The data buffer size is chosen based on the tape's * At most, there is only one ide-tape originated data transfer request
* recommendation. * in the device request queue. This allows ide.c to easily service
* requests from the other device when we postpone our active request.
* In the pipelined operation mode, we use our internal pipeline
* structure to hold more data requests. The data buffer size is chosen
* based on the tape's recommendation.
*/ */
/* Pointer to the request which is waiting in the device request queue */ /* ptr to the request which is waiting in the device request queue */
struct request *active_data_request; struct request *active_data_rq;
/* Data buffer size (chosen based on the tape's recommendation */ /* Data buffer size chosen based on the tape's recommendation */
int stage_size; int stage_size;
idetape_stage_t *merge_stage; idetape_stage_t *merge_stage;
int merge_stage_size; int merge_stage_size;
struct idetape_bh *bh; struct idetape_bh *bh;
char *b_data; char *b_data;
int b_count; int b_count;
/* /*
* Pipeline parameters. * Pipeline parameters.
* *
* To accomplish non-pipelined mode, we simply set the following * To accomplish non-pipelined mode, we simply set the following
* variables to zero (or NULL, where appropriate). * variables to zero (or NULL, where appropriate).
*/ */
/* Number of currently used stages */ /* Number of currently used stages */
int nr_stages; int nr_stages;
...@@ -378,20 +360,13 @@ typedef struct ide_tape_obj { ...@@ -378,20 +360,13 @@ typedef struct ide_tape_obj {
/* Status/Action flags: long for set_bit */ /* Status/Action flags: long for set_bit */
unsigned long flags; unsigned long flags;
/* protects the ide-tape queue */ /* protects the ide-tape queue */
spinlock_t spinlock; spinlock_t lock;
/* /* Measures average tape speed */
* Measures average tape speed
*/
unsigned long avg_time; unsigned long avg_time;
int avg_size; int avg_size;
int avg_speed; int avg_speed;
char vendor_id[10];
char product_id[18];
char firmware_revision[6];
int firmware_revision_num;
/* the door is currently locked */ /* the door is currently locked */
int door_locked; int door_locked;
/* the tape hardware is write protected */ /* the tape hardware is write protected */
...@@ -400,11 +375,9 @@ typedef struct ide_tape_obj { ...@@ -400,11 +375,9 @@ typedef struct ide_tape_obj {
char write_prot; char write_prot;
/* /*
* Limit the number of times a request can * Limit the number of times a request can be postponed, to avoid an
* be postponed, to avoid an infinite postpone * infinite postpone deadlock.
* deadlock.
*/ */
/* request postpone count limit */
int postpone_cnt; int postpone_cnt;
/* /*
...@@ -419,30 +392,19 @@ typedef struct ide_tape_obj { ...@@ -419,30 +392,19 @@ typedef struct ide_tape_obj {
int tape_head; int tape_head;
int last_tape_head; int last_tape_head;
/* /* Speed control at the tape buffers input/output */
* Speed control at the tape buffers input/output
*/
unsigned long insert_time; unsigned long insert_time;
int insert_size; int insert_size;
int insert_speed; int insert_speed;
int max_insert_speed; int max_insert_speed;
int measure_insert_time; int measure_insert_time;
/* /* Speed regulation negative feedback loop */
* Measure tape still time, in milliseconds
*/
unsigned long tape_still_time_begin;
int tape_still_time;
/*
* Speed regulation negative feedback loop
*/
int speed_control; int speed_control;
int pipeline_head_speed; int pipeline_head_speed;
int controlled_pipeline_head_speed; int controlled_pipeline_head_speed;
int uncontrolled_pipeline_head_speed; int uncontrolled_pipeline_head_speed;
int controlled_last_pipeline_head; int controlled_last_pipeline_head;
int uncontrolled_last_pipeline_head;
unsigned long uncontrolled_pipeline_head_time; unsigned long uncontrolled_pipeline_head_time;
unsigned long controlled_pipeline_head_time; unsigned long controlled_pipeline_head_time;
int controlled_previous_pipeline_head; int controlled_previous_pipeline_head;
...@@ -451,18 +413,7 @@ typedef struct ide_tape_obj { ...@@ -451,18 +413,7 @@ typedef struct ide_tape_obj {
unsigned long uncontrolled_previous_head_time; unsigned long uncontrolled_previous_head_time;
int restart_speed_control_req; int restart_speed_control_req;
/* u32 debug_mask;
* Debug_level determines amount of debugging output;
* can be changed using /proc/ide/hdx/settings
* 0 : almost no debugging output
* 1 : 0+output errors only
* 2 : 1+output all sensekey/asc
* 3 : 2+follow all chrdev related procedures
* 4 : 3+follow all procedures
* 5 : 4+include pc_stack rq_stack info
* 6 : 5+USE_COUNT updates
*/
int debug_level;
} idetape_tape_t; } idetape_tape_t;
static DEFINE_MUTEX(idetape_ref_mutex); static DEFINE_MUTEX(idetape_ref_mutex);
...@@ -495,9 +446,7 @@ static void ide_tape_put(struct ide_tape_obj *tape) ...@@ -495,9 +446,7 @@ static void ide_tape_put(struct ide_tape_obj *tape)
mutex_unlock(&idetape_ref_mutex); mutex_unlock(&idetape_ref_mutex);
} }
/* /* Tape door status */
* Tape door status
*/
#define DOOR_UNLOCKED 0 #define DOOR_UNLOCKED 0
#define DOOR_LOCKED 1 #define DOOR_LOCKED 1
#define DOOR_EXPLICITLY_LOCKED 2 #define DOOR_EXPLICITLY_LOCKED 2
...@@ -517,30 +466,23 @@ static void ide_tape_put(struct ide_tape_obj *tape) ...@@ -517,30 +466,23 @@ static void ide_tape_put(struct ide_tape_obj *tape)
/* 0 = no tape is loaded, so we don't rewind after ejecting */ /* 0 = no tape is loaded, so we don't rewind after ejecting */
#define IDETAPE_MEDIUM_PRESENT 9 #define IDETAPE_MEDIUM_PRESENT 9
/* /* A define for the READ BUFFER command */
* Some defines for the READ BUFFER command
*/
#define IDETAPE_RETRIEVE_FAULTY_BLOCK 6 #define IDETAPE_RETRIEVE_FAULTY_BLOCK 6
/* /* Some defines for the SPACE command */
* Some defines for the SPACE command
*/
#define IDETAPE_SPACE_OVER_FILEMARK 1 #define IDETAPE_SPACE_OVER_FILEMARK 1
#define IDETAPE_SPACE_TO_EOD 3 #define IDETAPE_SPACE_TO_EOD 3
/* /* Some defines for the LOAD UNLOAD command */
* Some defines for the LOAD UNLOAD command
*/
#define IDETAPE_LU_LOAD_MASK 1 #define IDETAPE_LU_LOAD_MASK 1
#define IDETAPE_LU_RETENSION_MASK 2 #define IDETAPE_LU_RETENSION_MASK 2
#define IDETAPE_LU_EOT_MASK 4 #define IDETAPE_LU_EOT_MASK 4
/* /*
* Special requests for our block device strategy routine. * Special requests for our block device strategy routine.
* *
* In order to service a character device command, we add special * In order to service a character device command, we add special requests to
* requests to the tail of our block device request queue and wait * the tail of our block device request queue and wait for their completion.
* for their completion.
*/ */
enum { enum {
...@@ -551,55 +493,20 @@ enum { ...@@ -551,55 +493,20 @@ enum {
REQ_IDETAPE_READ_BUFFER = (1 << 4), REQ_IDETAPE_READ_BUFFER = (1 << 4),
}; };
/* /* Error codes returned in rq->errors to the higher part of the driver. */
* Error codes which are returned in rq->errors to the higher part
* of the driver.
*/
#define IDETAPE_ERROR_GENERAL 101 #define IDETAPE_ERROR_GENERAL 101
#define IDETAPE_ERROR_FILEMARK 102 #define IDETAPE_ERROR_FILEMARK 102
#define IDETAPE_ERROR_EOD 103 #define IDETAPE_ERROR_EOD 103
/*
* The following is used to format the general configuration word of
* the ATAPI IDENTIFY DEVICE command.
*/
struct idetape_id_gcw {
unsigned packet_size :2; /* Packet Size */
unsigned reserved234 :3; /* Reserved */
unsigned drq_type :2; /* Command packet DRQ type */
unsigned removable :1; /* Removable media */
unsigned device_type :5; /* Device type */
unsigned reserved13 :1; /* Reserved */
unsigned protocol :2; /* Protocol type */
};
/*
* READ POSITION packet command - Data Format (From Table 6-57)
*/
typedef struct {
unsigned reserved0_10 :2; /* Reserved */
unsigned bpu :1; /* Block Position Unknown */
unsigned reserved0_543 :3; /* Reserved */
unsigned eop :1; /* End Of Partition */
unsigned bop :1; /* Beginning Of Partition */
u8 partition; /* Partition Number */
u8 reserved2, reserved3; /* Reserved */
u32 first_block; /* First Block Location */
u32 last_block; /* Last Block Location (Optional) */
u8 reserved12; /* Reserved */
u8 blocks_in_buffer[3]; /* Blocks In Buffer - (Optional) */
u32 bytes_in_buffer; /* Bytes In Buffer (Optional) */
} idetape_read_position_result_t;
/* Structures related to the SELECT SENSE / MODE SENSE packet commands. */ /* Structures related to the SELECT SENSE / MODE SENSE packet commands. */
#define IDETAPE_BLOCK_DESCRIPTOR 0 #define IDETAPE_BLOCK_DESCRIPTOR 0
#define IDETAPE_CAPABILITIES_PAGE 0x2a #define IDETAPE_CAPABILITIES_PAGE 0x2a
/* /*
* The variables below are used for the character device interface. * The variables below are used for the character device interface. Additional
* Additional state variables are defined in our ide_drive_t structure. * state variables are defined in our ide_drive_t structure.
*/ */
static struct ide_tape_obj * idetape_devs[MAX_HWIFS * MAX_DRIVES]; static struct ide_tape_obj *idetape_devs[MAX_HWIFS * MAX_DRIVES];
#define ide_tape_f(file) ((file)->private_data) #define ide_tape_f(file) ((file)->private_data)
...@@ -615,24 +522,18 @@ static struct ide_tape_obj *ide_tape_chrdev_get(unsigned int i) ...@@ -615,24 +522,18 @@ static struct ide_tape_obj *ide_tape_chrdev_get(unsigned int i)
return tape; return tape;
} }
/*
* Function declarations
*
*/
static int idetape_chrdev_release (struct inode *inode, struct file *filp);
static void idetape_write_release (ide_drive_t *drive, unsigned int minor);
/* /*
* Too bad. The drive wants to send us data which we are not ready to accept. * Too bad. The drive wants to send us data which we are not ready to accept.
* Just throw it away. * Just throw it away.
*/ */
static void idetape_discard_data (ide_drive_t *drive, unsigned int bcount) static void idetape_discard_data(ide_drive_t *drive, unsigned int bcount)
{ {
while (bcount--) while (bcount--)
(void) HWIF(drive)->INB(IDE_DATA_REG); (void) HWIF(drive)->INB(IDE_DATA_REG);
} }
static void idetape_input_buffers (ide_drive_t *drive, idetape_pc_t *pc, unsigned int bcount) static void idetape_input_buffers(ide_drive_t *drive, idetape_pc_t *pc,
unsigned int bcount)
{ {
struct idetape_bh *bh = pc->bh; struct idetape_bh *bh = pc->bh;
int count; int count;
...@@ -644,8 +545,11 @@ static void idetape_input_buffers (ide_drive_t *drive, idetape_pc_t *pc, unsigne ...@@ -644,8 +545,11 @@ static void idetape_input_buffers (ide_drive_t *drive, idetape_pc_t *pc, unsigne
idetape_discard_data(drive, bcount); idetape_discard_data(drive, bcount);
return; return;
} }
count = min((unsigned int)(bh->b_size - atomic_read(&bh->b_count)), bcount); count = min(
HWIF(drive)->atapi_input_bytes(drive, bh->b_data + atomic_read(&bh->b_count), count); (unsigned int)(bh->b_size - atomic_read(&bh->b_count)),
bcount);
HWIF(drive)->atapi_input_bytes(drive, bh->b_data +
atomic_read(&bh->b_count), count);
bcount -= count; bcount -= count;
atomic_add(count, &bh->b_count); atomic_add(count, &bh->b_count);
if (atomic_read(&bh->b_count) == bh->b_size) { if (atomic_read(&bh->b_count) == bh->b_size) {
...@@ -657,15 +561,16 @@ static void idetape_input_buffers (ide_drive_t *drive, idetape_pc_t *pc, unsigne ...@@ -657,15 +561,16 @@ static void idetape_input_buffers (ide_drive_t *drive, idetape_pc_t *pc, unsigne
pc->bh = bh; pc->bh = bh;
} }
static void idetape_output_buffers (ide_drive_t *drive, idetape_pc_t *pc, unsigned int bcount) static void idetape_output_buffers(ide_drive_t *drive, idetape_pc_t *pc,
unsigned int bcount)
{ {
struct idetape_bh *bh = pc->bh; struct idetape_bh *bh = pc->bh;
int count; int count;
while (bcount) { while (bcount) {
if (bh == NULL) { if (bh == NULL) {
printk(KERN_ERR "ide-tape: bh == NULL in " printk(KERN_ERR "ide-tape: bh == NULL in %s\n",
"idetape_output_buffers\n"); __func__);
return; return;
} }
count = min((unsigned int)pc->b_count, (unsigned int)bcount); count = min((unsigned int)pc->b_count, (unsigned int)bcount);
...@@ -674,7 +579,8 @@ static void idetape_output_buffers (ide_drive_t *drive, idetape_pc_t *pc, unsign ...@@ -674,7 +579,8 @@ static void idetape_output_buffers (ide_drive_t *drive, idetape_pc_t *pc, unsign
pc->b_data += count; pc->b_data += count;
pc->b_count -= count; pc->b_count -= count;
if (!pc->b_count) { if (!pc->b_count) {
pc->bh = bh = bh->b_reqnext; bh = bh->b_reqnext;
pc->bh = bh;
if (bh) { if (bh) {
pc->b_data = bh->b_data; pc->b_data = bh->b_data;
pc->b_count = atomic_read(&bh->b_count); pc->b_count = atomic_read(&bh->b_count);
...@@ -683,7 +589,7 @@ static void idetape_output_buffers (ide_drive_t *drive, idetape_pc_t *pc, unsign ...@@ -683,7 +589,7 @@ static void idetape_output_buffers (ide_drive_t *drive, idetape_pc_t *pc, unsign
} }
} }
static void idetape_update_buffers (idetape_pc_t *pc) static void idetape_update_buffers(idetape_pc_t *pc)
{ {
struct idetape_bh *bh = pc->bh; struct idetape_bh *bh = pc->bh;
int count; int count;
...@@ -693,8 +599,8 @@ static void idetape_update_buffers (idetape_pc_t *pc) ...@@ -693,8 +599,8 @@ static void idetape_update_buffers (idetape_pc_t *pc)
return; return;
while (bcount) { while (bcount) {
if (bh == NULL) { if (bh == NULL) {
printk(KERN_ERR "ide-tape: bh == NULL in " printk(KERN_ERR "ide-tape: bh == NULL in %s\n",
"idetape_update_buffers\n"); __func__);
return; return;
} }
count = min((unsigned int)bh->b_size, (unsigned int)bcount); count = min((unsigned int)bh->b_size, (unsigned int)bcount);
...@@ -712,17 +618,14 @@ static void idetape_update_buffers (idetape_pc_t *pc) ...@@ -712,17 +618,14 @@ static void idetape_update_buffers (idetape_pc_t *pc)
* driver. A storage space for a maximum of IDETAPE_PC_STACK packet * driver. A storage space for a maximum of IDETAPE_PC_STACK packet
* commands is allocated at initialization time. * commands is allocated at initialization time.
*/ */
static idetape_pc_t *idetape_next_pc_storage (ide_drive_t *drive) static idetape_pc_t *idetape_next_pc_storage(ide_drive_t *drive)
{ {
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
#if IDETAPE_DEBUG_LOG debug_log(DBG_PCRQ_STACK, "pc_stack_index=%d\n", tape->pc_stack_index);
if (tape->debug_level >= 5)
printk(KERN_INFO "ide-tape: pc_stack_index=%d\n",
tape->pc_stack_index);
#endif /* IDETAPE_DEBUG_LOG */
if (tape->pc_stack_index == IDETAPE_PC_STACK) if (tape->pc_stack_index == IDETAPE_PC_STACK)
tape->pc_stack_index=0; tape->pc_stack_index = 0;
return (&tape->pc_stack[tape->pc_stack_index++]); return (&tape->pc_stack[tape->pc_stack_index++]);
} }
...@@ -731,32 +634,26 @@ static idetape_pc_t *idetape_next_pc_storage (ide_drive_t *drive) ...@@ -731,32 +634,26 @@ static idetape_pc_t *idetape_next_pc_storage (ide_drive_t *drive)
* Since we queue packet commands in the request queue, we need to * Since we queue packet commands in the request queue, we need to
* allocate a request, along with the allocation of a packet command. * allocate a request, along with the allocation of a packet command.
*/ */
/************************************************************** /**************************************************************
* * * *
* This should get fixed to use kmalloc(.., GFP_ATOMIC) * * This should get fixed to use kmalloc(.., GFP_ATOMIC) *
* followed later on by kfree(). -ml * * followed later on by kfree(). -ml *
* * * *
**************************************************************/ **************************************************************/
static struct request *idetape_next_rq_storage (ide_drive_t *drive) static struct request *idetape_next_rq_storage(ide_drive_t *drive)
{ {
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
#if IDETAPE_DEBUG_LOG debug_log(DBG_PCRQ_STACK, "rq_stack_index=%d\n", tape->rq_stack_index);
if (tape->debug_level >= 5)
printk(KERN_INFO "ide-tape: rq_stack_index=%d\n",
tape->rq_stack_index);
#endif /* IDETAPE_DEBUG_LOG */
if (tape->rq_stack_index == IDETAPE_PC_STACK) if (tape->rq_stack_index == IDETAPE_PC_STACK)
tape->rq_stack_index=0; tape->rq_stack_index = 0;
return (&tape->rq_stack[tape->rq_stack_index++]); return (&tape->rq_stack[tape->rq_stack_index++]);
} }
/* static void idetape_init_pc(idetape_pc_t *pc)
* idetape_init_pc initializes a packet command.
*/
static void idetape_init_pc (idetape_pc_t *pc)
{ {
memset(pc->c, 0, 12); memset(pc->c, 0, 12);
pc->retries = 0; pc->retries = 0;
...@@ -780,22 +677,14 @@ static void idetape_analyze_error(ide_drive_t *drive, u8 *sense) ...@@ -780,22 +677,14 @@ static void idetape_analyze_error(ide_drive_t *drive, u8 *sense)
tape->sense_key = sense[2] & 0xF; tape->sense_key = sense[2] & 0xF;
tape->asc = sense[12]; tape->asc = sense[12];
tape->ascq = sense[13]; tape->ascq = sense[13];
#if IDETAPE_DEBUG_LOG
/* debug_log(DBG_ERR, "pc = %x, sense key = %x, asc = %x, ascq = %x\n",
* Without debugging, we only log an error if we decided to give up pc->c[0], tape->sense_key, tape->asc, tape->ascq);
* retrying.
*/
if (tape->debug_level >= 1)
printk(KERN_INFO "ide-tape: pc = %x, sense key = %x, "
"asc = %x, ascq = %x\n",
pc->c[0], tape->sense_key,
tape->asc, tape->ascq);
#endif /* IDETAPE_DEBUG_LOG */
/* Correct pc->actually_transferred by asking the tape. */ /* Correct pc->actually_transferred by asking the tape. */
if (test_bit(PC_DMA_ERROR, &pc->flags)) { if (test_bit(PC_DMA_ERROR, &pc->flags)) {
pc->actually_transferred = pc->request_transfer - pc->actually_transferred = pc->request_transfer -
tape->tape_block_size * tape->blk_size *
be32_to_cpu(get_unaligned((u32 *)&sense[3])); be32_to_cpu(get_unaligned((u32 *)&sense[3]));
idetape_update_buffers(pc); idetape_update_buffers(pc);
} }
...@@ -843,50 +732,24 @@ static void idetape_activate_next_stage(ide_drive_t *drive) ...@@ -843,50 +732,24 @@ static void idetape_activate_next_stage(ide_drive_t *drive)
idetape_stage_t *stage = tape->next_stage; idetape_stage_t *stage = tape->next_stage;
struct request *rq = &stage->rq; struct request *rq = &stage->rq;
#if IDETAPE_DEBUG_LOG debug_log(DBG_PROCS, "Enter %s\n", __func__);
if (tape->debug_level >= 4)
printk(KERN_INFO "ide-tape: Reached idetape_active_next_stage\n");
#endif /* IDETAPE_DEBUG_LOG */
if (stage == NULL) { if (stage == NULL) {
printk(KERN_ERR "ide-tape: bug: Trying to activate a non existing stage\n"); printk(KERN_ERR "ide-tape: bug: Trying to activate a non"
" existing stage\n");
return; return;
} }
rq->rq_disk = tape->disk; rq->rq_disk = tape->disk;
rq->buffer = NULL; rq->buffer = NULL;
rq->special = (void *)stage->bh; rq->special = (void *)stage->bh;
tape->active_data_request = rq; tape->active_data_rq = rq;
tape->active_stage = stage; tape->active_stage = stage;
tape->next_stage = stage->next; tape->next_stage = stage->next;
} }
/* /* Free a stage along with its related buffers completely. */
* idetape_increase_max_pipeline_stages is a part of the feedback static void __idetape_kfree_stage(idetape_stage_t *stage)
* loop which tries to find the optimum number of stages. In the
* feedback loop, we are starting from a minimum maximum number of
* stages, and if we sense that the pipeline is empty, we try to
* increase it, until we reach the user compile time memory limit.
*/
static void idetape_increase_max_pipeline_stages (ide_drive_t *drive)
{
idetape_tape_t *tape = drive->driver_data;
int increase = (tape->max_pipeline - tape->min_pipeline) / 10;
#if IDETAPE_DEBUG_LOG
if (tape->debug_level >= 4)
printk (KERN_INFO "ide-tape: Reached idetape_increase_max_pipeline_stages\n");
#endif /* IDETAPE_DEBUG_LOG */
tape->max_stages += max(increase, 1);
tape->max_stages = max(tape->max_stages, tape->min_pipeline);
tape->max_stages = min(tape->max_stages, tape->max_pipeline);
}
/*
* idetape_kfree_stage calls kfree to completely free a stage, along with
* its related buffers.
*/
static void __idetape_kfree_stage (idetape_stage_t *stage)
{ {
struct idetape_bh *prev_bh, *bh = stage->bh; struct idetape_bh *prev_bh, *bh = stage->bh;
int size; int size;
...@@ -907,30 +770,29 @@ static void __idetape_kfree_stage (idetape_stage_t *stage) ...@@ -907,30 +770,29 @@ static void __idetape_kfree_stage (idetape_stage_t *stage)
kfree(stage); kfree(stage);
} }
static void idetape_kfree_stage (idetape_tape_t *tape, idetape_stage_t *stage) static void idetape_kfree_stage(idetape_tape_t *tape, idetape_stage_t *stage)
{ {
__idetape_kfree_stage(stage); __idetape_kfree_stage(stage);
} }
/* /*
* idetape_remove_stage_head removes tape->first_stage from the pipeline. * Remove tape->first_stage from the pipeline. The caller should avoid race
* The caller should avoid race conditions. * conditions.
*/ */
static void idetape_remove_stage_head (ide_drive_t *drive) static void idetape_remove_stage_head(ide_drive_t *drive)
{ {
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
idetape_stage_t *stage; idetape_stage_t *stage;
#if IDETAPE_DEBUG_LOG debug_log(DBG_PROCS, "Enter %s\n", __func__);
if (tape->debug_level >= 4)
printk(KERN_INFO "ide-tape: Reached idetape_remove_stage_head\n");
#endif /* IDETAPE_DEBUG_LOG */
if (tape->first_stage == NULL) { if (tape->first_stage == NULL) {
printk(KERN_ERR "ide-tape: bug: tape->first_stage is NULL\n"); printk(KERN_ERR "ide-tape: bug: tape->first_stage is NULL\n");
return; return;
} }
if (tape->active_stage == tape->first_stage) { if (tape->active_stage == tape->first_stage) {
printk(KERN_ERR "ide-tape: bug: Trying to free our active pipeline stage\n"); printk(KERN_ERR "ide-tape: bug: Trying to free our active "
"pipeline stage\n");
return; return;
} }
stage = tape->first_stage; stage = tape->first_stage;
...@@ -940,9 +802,11 @@ static void idetape_remove_stage_head (ide_drive_t *drive) ...@@ -940,9 +802,11 @@ static void idetape_remove_stage_head (ide_drive_t *drive)
if (tape->first_stage == NULL) { if (tape->first_stage == NULL) {
tape->last_stage = NULL; tape->last_stage = NULL;
if (tape->next_stage != NULL) if (tape->next_stage != NULL)
printk(KERN_ERR "ide-tape: bug: tape->next_stage != NULL\n"); printk(KERN_ERR "ide-tape: bug: tape->next_stage !="
" NULL\n");
if (tape->nr_stages) if (tape->nr_stages)
printk(KERN_ERR "ide-tape: bug: nr_stages should be 0 now\n"); printk(KERN_ERR "ide-tape: bug: nr_stages should be 0 "
"now\n");
} }
} }
...@@ -957,10 +821,8 @@ static void idetape_abort_pipeline(ide_drive_t *drive, ...@@ -957,10 +821,8 @@ static void idetape_abort_pipeline(ide_drive_t *drive,
idetape_stage_t *stage = new_last_stage->next; idetape_stage_t *stage = new_last_stage->next;
idetape_stage_t *nstage; idetape_stage_t *nstage;
#if IDETAPE_DEBUG_LOG debug_log(DBG_PROCS, "%s: Enter %s\n", tape->name, __func__);
if (tape->debug_level >= 4)
printk(KERN_INFO "ide-tape: %s: idetape_abort_pipeline called\n", tape->name);
#endif
while (stage) { while (stage) {
nstage = stage->next; nstage = stage->next;
idetape_kfree_stage(tape, stage); idetape_kfree_stage(tape, stage);
...@@ -975,8 +837,8 @@ static void idetape_abort_pipeline(ide_drive_t *drive, ...@@ -975,8 +837,8 @@ static void idetape_abort_pipeline(ide_drive_t *drive,
} }
/* /*
* idetape_end_request is used to finish servicing a request, and to * Finish servicing a request and insert a pending pipeline request into the
* insert a pending pipeline request into the main device queue. * main device queue.
*/ */
static int idetape_end_request(ide_drive_t *drive, int uptodate, int nr_sects) static int idetape_end_request(ide_drive_t *drive, int uptodate, int nr_sects)
{ {
...@@ -987,15 +849,12 @@ static int idetape_end_request(ide_drive_t *drive, int uptodate, int nr_sects) ...@@ -987,15 +849,12 @@ static int idetape_end_request(ide_drive_t *drive, int uptodate, int nr_sects)
int remove_stage = 0; int remove_stage = 0;
idetape_stage_t *active_stage; idetape_stage_t *active_stage;
#if IDETAPE_DEBUG_LOG debug_log(DBG_PROCS, "Enter %s\n", __func__);
if (tape->debug_level >= 4)
printk(KERN_INFO "ide-tape: Reached idetape_end_request\n");
#endif /* IDETAPE_DEBUG_LOG */
switch (uptodate) { switch (uptodate) {
case 0: error = IDETAPE_ERROR_GENERAL; break; case 0: error = IDETAPE_ERROR_GENERAL; break;
case 1: error = 0; break; case 1: error = 0; break;
default: error = uptodate; default: error = uptodate;
} }
rq->errors = error; rq->errors = error;
if (error) if (error)
...@@ -1006,20 +865,21 @@ static int idetape_end_request(ide_drive_t *drive, int uptodate, int nr_sects) ...@@ -1006,20 +865,21 @@ static int idetape_end_request(ide_drive_t *drive, int uptodate, int nr_sects)
return 0; return 0;
} }
spin_lock_irqsave(&tape->spinlock, flags); spin_lock_irqsave(&tape->lock, flags);
/* The request was a pipelined data transfer request */ /* The request was a pipelined data transfer request */
if (tape->active_data_request == rq) { if (tape->active_data_rq == rq) {
active_stage = tape->active_stage; active_stage = tape->active_stage;
tape->active_stage = NULL; tape->active_stage = NULL;
tape->active_data_request = NULL; tape->active_data_rq = NULL;
tape->nr_pending_stages--; tape->nr_pending_stages--;
if (rq->cmd[0] & REQ_IDETAPE_WRITE) { if (rq->cmd[0] & REQ_IDETAPE_WRITE) {
remove_stage = 1; remove_stage = 1;
if (error) { if (error) {
set_bit(IDETAPE_PIPELINE_ERROR, &tape->flags); set_bit(IDETAPE_PIPELINE_ERROR, &tape->flags);
if (error == IDETAPE_ERROR_EOD) if (error == IDETAPE_ERROR_EOD)
idetape_abort_pipeline(drive, active_stage); idetape_abort_pipeline(drive,
active_stage);
} }
} else if (rq->cmd[0] & REQ_IDETAPE_READ) { } else if (rq->cmd[0] & REQ_IDETAPE_READ) {
if (error == IDETAPE_ERROR_EOD) { if (error == IDETAPE_ERROR_EOD) {
...@@ -1030,48 +890,57 @@ static int idetape_end_request(ide_drive_t *drive, int uptodate, int nr_sects) ...@@ -1030,48 +890,57 @@ static int idetape_end_request(ide_drive_t *drive, int uptodate, int nr_sects)
if (tape->next_stage != NULL) { if (tape->next_stage != NULL) {
idetape_activate_next_stage(drive); idetape_activate_next_stage(drive);
/* Insert the next request into the request queue. */
(void)ide_do_drive_cmd(drive, tape->active_data_rq,
ide_end);
} else if (!error) {
/* /*
* Insert the next request into the request queue. * This is a part of the feedback loop which tries to
* find the optimum number of stages. We are starting
* from a minimum maximum number of stages, and if we
* sense that the pipeline is empty, we try to increase
* it, until we reach the user compile time memory
* limit.
*/ */
(void) ide_do_drive_cmd(drive, tape->active_data_request, ide_end); int i = (tape->max_pipeline - tape->min_pipeline) / 10;
} else if (!error) {
idetape_increase_max_pipeline_stages(drive); tape->max_stages += max(i, 1);
tape->max_stages = max(tape->max_stages,
tape->min_pipeline);
tape->max_stages = min(tape->max_stages,
tape->max_pipeline);
} }
} }
ide_end_drive_cmd(drive, 0, 0); ide_end_drive_cmd(drive, 0, 0);
// blkdev_dequeue_request(rq);
// drive->rq = NULL;
// end_that_request_last(rq);
if (remove_stage) if (remove_stage)
idetape_remove_stage_head(drive); idetape_remove_stage_head(drive);
if (tape->active_data_request == NULL) if (tape->active_data_rq == NULL)
clear_bit(IDETAPE_PIPELINE_ACTIVE, &tape->flags); clear_bit(IDETAPE_PIPELINE_ACTIVE, &tape->flags);
spin_unlock_irqrestore(&tape->spinlock, flags); spin_unlock_irqrestore(&tape->lock, flags);
return 0; return 0;
} }
static ide_startstop_t idetape_request_sense_callback (ide_drive_t *drive) static ide_startstop_t idetape_request_sense_callback(ide_drive_t *drive)
{ {
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
#if IDETAPE_DEBUG_LOG debug_log(DBG_PROCS, "Enter %s\n", __func__);
if (tape->debug_level >= 4)
printk(KERN_INFO "ide-tape: Reached idetape_request_sense_callback\n");
#endif /* IDETAPE_DEBUG_LOG */
if (!tape->pc->error) { if (!tape->pc->error) {
idetape_analyze_error(drive, tape->pc->buffer); idetape_analyze_error(drive, tape->pc->buffer);
idetape_end_request(drive, 1, 0); idetape_end_request(drive, 1, 0);
} else { } else {
printk(KERN_ERR "ide-tape: Error in REQUEST SENSE itself - Aborting request!\n"); printk(KERN_ERR "ide-tape: Error in REQUEST SENSE itself - "
"Aborting request!\n");
idetape_end_request(drive, 0, 0); idetape_end_request(drive, 0, 0);
} }
return ide_stopped; return ide_stopped;
} }
static void idetape_create_request_sense_cmd (idetape_pc_t *pc) static void idetape_create_request_sense_cmd(idetape_pc_t *pc)
{ {
idetape_init_pc(pc); idetape_init_pc(pc);
pc->c[0] = REQUEST_SENSE; pc->c[0] = REQUEST_SENSE;
pc->c[4] = 20; pc->c[4] = 20;
pc->request_transfer = 20; pc->request_transfer = 20;
...@@ -1086,25 +955,22 @@ static void idetape_init_rq(struct request *rq, u8 cmd) ...@@ -1086,25 +955,22 @@ static void idetape_init_rq(struct request *rq, u8 cmd)
} }
/* /*
* idetape_queue_pc_head generates a new packet command request in front * Generate a new packet command request in front of the request queue, before
* of the request queue, before the current request, so that it will be * the current request, so that it will be processed immediately, on the next
* processed immediately, on the next pass through the driver. * pass through the driver. The function below is called from the request
* * handling part of the driver (the "bottom" part). Safe storage for the request
* idetape_queue_pc_head is called from the request handling part of * should be allocated with ide_tape_next_{pc,rq}_storage() prior to that.
* the driver (the "bottom" part). Safe storage for the request should
* be allocated with idetape_next_pc_storage and idetape_next_rq_storage
* before calling idetape_queue_pc_head.
* *
* Memory for those requests is pre-allocated at initialization time, and * Memory for those requests is pre-allocated at initialization time, and is
* is limited to IDETAPE_PC_STACK requests. We assume that we have enough * limited to IDETAPE_PC_STACK requests. We assume that we have enough space for
* space for the maximum possible number of inter-dependent packet commands. * the maximum possible number of inter-dependent packet commands.
* *
* The higher level of the driver - The ioctl handler and the character * The higher level of the driver - The ioctl handler and the character device
* device handling functions should queue request to the lower level part * handling functions should queue request to the lower level part and wait for
* and wait for their completion using idetape_queue_pc_tail or * their completion using idetape_queue_pc_tail or idetape_queue_rw_tail.
* idetape_queue_rw_tail.
*/ */
static void idetape_queue_pc_head (ide_drive_t *drive, idetape_pc_t *pc,struct request *rq) static void idetape_queue_pc_head(ide_drive_t *drive, idetape_pc_t *pc,
struct request *rq)
{ {
struct ide_tape_obj *tape = drive->driver_data; struct ide_tape_obj *tape = drive->driver_data;
...@@ -1125,7 +991,7 @@ static ide_startstop_t idetape_retry_pc (ide_drive_t *drive) ...@@ -1125,7 +991,7 @@ static ide_startstop_t idetape_retry_pc (ide_drive_t *drive)
idetape_pc_t *pc; idetape_pc_t *pc;
struct request *rq; struct request *rq;
(void)drive->hwif->INB(IDE_ERROR_REG); (void)ide_read_error(drive);
pc = idetape_next_pc_storage(drive); pc = idetape_next_pc_storage(drive);
rq = idetape_next_rq_storage(drive); rq = idetape_next_rq_storage(drive);
idetape_create_request_sense_cmd(pc); idetape_create_request_sense_cmd(pc);
...@@ -1135,50 +1001,46 @@ static ide_startstop_t idetape_retry_pc (ide_drive_t *drive) ...@@ -1135,50 +1001,46 @@ static ide_startstop_t idetape_retry_pc (ide_drive_t *drive)
} }
/* /*
* idetape_postpone_request postpones the current request so that * Postpone the current request so that ide.c will be able to service requests
* ide.c will be able to service requests from another device on * from another device on the same hwgroup while we are polling for DSC.
* the same hwgroup while we are polling for DSC.
*/ */
static void idetape_postpone_request (ide_drive_t *drive) static void idetape_postpone_request(ide_drive_t *drive)
{ {
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
#if IDETAPE_DEBUG_LOG debug_log(DBG_PROCS, "Enter %s\n", __func__);
if (tape->debug_level >= 4)
printk(KERN_INFO "ide-tape: idetape_postpone_request\n");
#endif
tape->postponed_rq = HWGROUP(drive)->rq; tape->postponed_rq = HWGROUP(drive)->rq;
ide_stall_queue(drive, tape->dsc_polling_frequency); ide_stall_queue(drive, tape->dsc_poll_freq);
} }
typedef void idetape_io_buf(ide_drive_t *, idetape_pc_t *, unsigned int);
/* /*
* idetape_pc_intr is the usual interrupt handler which will be called * This is the usual interrupt handler which will be called during a packet
* during a packet command. We will transfer some of the data (as * command. We will transfer some of the data (as requested by the drive) and
* requested by the drive) and will re-point interrupt handler to us. * will re-point interrupt handler to us. When data transfer is finished, we
* When data transfer is finished, we will act according to the * will act according to the algorithm described before
* algorithm described before idetape_issue_packet_command. * idetape_issue_pc.
*
*/ */
static ide_startstop_t idetape_pc_intr (ide_drive_t *drive) static ide_startstop_t idetape_pc_intr(ide_drive_t *drive)
{ {
ide_hwif_t *hwif = drive->hwif; ide_hwif_t *hwif = drive->hwif;
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
idetape_pc_t *pc = tape->pc; idetape_pc_t *pc = tape->pc;
xfer_func_t *xferfunc;
idetape_io_buf *iobuf;
unsigned int temp; unsigned int temp;
#if SIMULATE_ERRORS #if SIMULATE_ERRORS
static int error_sim_count = 0; static int error_sim_count;
#endif #endif
u16 bcount; u16 bcount;
u8 stat, ireason; u8 stat, ireason;
#if IDETAPE_DEBUG_LOG debug_log(DBG_PROCS, "Enter %s - interrupt handler\n", __func__);
if (tape->debug_level >= 4)
printk(KERN_INFO "ide-tape: Reached idetape_pc_intr "
"interrupt handler\n");
#endif /* IDETAPE_DEBUG_LOG */
/* Clear the interrupt */ /* Clear the interrupt */
stat = hwif->INB(IDE_STATUS_REG); stat = ide_read_status(drive);
if (test_bit(PC_DMA_IN_PROGRESS, &pc->flags)) { if (test_bit(PC_DMA_IN_PROGRESS, &pc->flags)) {
if (hwif->ide_dma_end(drive) || (stat & ERR_STAT)) { if (hwif->ide_dma_end(drive) || (stat & ERR_STAT)) {
...@@ -1208,20 +1070,16 @@ static ide_startstop_t idetape_pc_intr (ide_drive_t *drive) ...@@ -1208,20 +1070,16 @@ static ide_startstop_t idetape_pc_intr (ide_drive_t *drive)
pc->actually_transferred = pc->request_transfer; pc->actually_transferred = pc->request_transfer;
idetape_update_buffers(pc); idetape_update_buffers(pc);
} }
#if IDETAPE_DEBUG_LOG debug_log(DBG_PROCS, "DMA finished\n");
if (tape->debug_level >= 4)
printk(KERN_INFO "ide-tape: DMA finished\n");
#endif /* IDETAPE_DEBUG_LOG */
} }
/* No more interrupts */ /* No more interrupts */
if ((stat & DRQ_STAT) == 0) { if ((stat & DRQ_STAT) == 0) {
#if IDETAPE_DEBUG_LOG debug_log(DBG_SENSE, "Packet command completed, %d bytes"
if (tape->debug_level >= 2) " transferred\n", pc->actually_transferred);
printk(KERN_INFO "ide-tape: Packet command completed, %d bytes transferred\n", pc->actually_transferred);
#endif /* IDETAPE_DEBUG_LOG */
clear_bit(PC_DMA_IN_PROGRESS, &pc->flags);
clear_bit(PC_DMA_IN_PROGRESS, &pc->flags);
local_irq_enable(); local_irq_enable();
#if SIMULATE_ERRORS #if SIMULATE_ERRORS
...@@ -1236,19 +1094,16 @@ static ide_startstop_t idetape_pc_intr (ide_drive_t *drive) ...@@ -1236,19 +1094,16 @@ static ide_startstop_t idetape_pc_intr (ide_drive_t *drive)
stat &= ~ERR_STAT; stat &= ~ERR_STAT;
if ((stat & ERR_STAT) || test_bit(PC_DMA_ERROR, &pc->flags)) { if ((stat & ERR_STAT) || test_bit(PC_DMA_ERROR, &pc->flags)) {
/* Error detected */ /* Error detected */
#if IDETAPE_DEBUG_LOG debug_log(DBG_ERR, "%s: I/O error\n", tape->name);
if (tape->debug_level >= 1)
printk(KERN_INFO "ide-tape: %s: I/O error\n",
tape->name);
#endif /* IDETAPE_DEBUG_LOG */
if (pc->c[0] == REQUEST_SENSE) { if (pc->c[0] == REQUEST_SENSE) {
printk(KERN_ERR "ide-tape: I/O error in request sense command\n"); printk(KERN_ERR "ide-tape: I/O error in request"
" sense command\n");
return ide_do_reset(drive); return ide_do_reset(drive);
} }
#if IDETAPE_DEBUG_LOG debug_log(DBG_ERR, "[cmd %x]: check condition\n",
if (tape->debug_level >= 1) pc->c[0]);
printk(KERN_INFO "ide-tape: [cmd %x]: check condition\n", pc->c[0]);
#endif
/* Retry operation */ /* Retry operation */
return idetape_retry_pc(drive); return idetape_retry_pc(drive);
} }
...@@ -1257,7 +1112,7 @@ static ide_startstop_t idetape_pc_intr (ide_drive_t *drive) ...@@ -1257,7 +1112,7 @@ static ide_startstop_t idetape_pc_intr (ide_drive_t *drive)
(stat & SEEK_STAT) == 0) { (stat & SEEK_STAT) == 0) {
/* Media access command */ /* Media access command */
tape->dsc_polling_start = jiffies; tape->dsc_polling_start = jiffies;
tape->dsc_polling_frequency = IDETAPE_DSC_MA_FAST; tape->dsc_poll_freq = IDETAPE_DSC_MA_FAST;
tape->dsc_timeout = jiffies + IDETAPE_DSC_MA_TIMEOUT; tape->dsc_timeout = jiffies + IDETAPE_DSC_MA_TIMEOUT;
/* Allow ide.c to handle other requests */ /* Allow ide.c to handle other requests */
idetape_postpone_request(drive); idetape_postpone_request(drive);
...@@ -1282,7 +1137,7 @@ static ide_startstop_t idetape_pc_intr (ide_drive_t *drive) ...@@ -1282,7 +1137,7 @@ static ide_startstop_t idetape_pc_intr (ide_drive_t *drive)
ireason = hwif->INB(IDE_IREASON_REG); ireason = hwif->INB(IDE_IREASON_REG);
if (ireason & CD) { if (ireason & CD) {
printk(KERN_ERR "ide-tape: CoD != 0 in idetape_pc_intr\n"); printk(KERN_ERR "ide-tape: CoD != 0 in %s\n", __func__);
return ide_do_reset(drive); return ide_do_reset(drive);
} }
if (((ireason & IO) == IO) == test_bit(PC_WRITING, &pc->flags)) { if (((ireason & IO) == IO) == test_bit(PC_WRITING, &pc->flags)) {
...@@ -1298,86 +1153,76 @@ static ide_startstop_t idetape_pc_intr (ide_drive_t *drive) ...@@ -1298,86 +1153,76 @@ static ide_startstop_t idetape_pc_intr (ide_drive_t *drive)
temp = pc->actually_transferred + bcount; temp = pc->actually_transferred + bcount;
if (temp > pc->request_transfer) { if (temp > pc->request_transfer) {
if (temp > pc->buffer_size) { if (temp > pc->buffer_size) {
printk(KERN_ERR "ide-tape: The tape wants to send us more data than expected - discarding data\n"); printk(KERN_ERR "ide-tape: The tape wants to "
"send us more data than expected "
"- discarding data\n");
idetape_discard_data(drive, bcount); idetape_discard_data(drive, bcount);
ide_set_handler(drive, &idetape_pc_intr, IDETAPE_WAIT_CMD, NULL); ide_set_handler(drive, &idetape_pc_intr,
IDETAPE_WAIT_CMD, NULL);
return ide_started; return ide_started;
} }
#if IDETAPE_DEBUG_LOG debug_log(DBG_SENSE, "The tape wants to send us more "
if (tape->debug_level >= 2) "data than expected - allowing transfer\n");
printk(KERN_NOTICE "ide-tape: The tape wants to send us more data than expected - allowing transfer\n");
#endif /* IDETAPE_DEBUG_LOG */
} }
} iobuf = &idetape_input_buffers;
if (test_bit(PC_WRITING, &pc->flags)) { xferfunc = hwif->atapi_input_bytes;
if (pc->bh != NULL)
idetape_output_buffers(drive, pc, bcount);
else
/* Write the current buffer */
hwif->atapi_output_bytes(drive, pc->current_position,
bcount);
} else { } else {
if (pc->bh != NULL) iobuf = &idetape_output_buffers;
idetape_input_buffers(drive, pc, bcount); xferfunc = hwif->atapi_output_bytes;
else
/* Read the current buffer */
hwif->atapi_input_bytes(drive, pc->current_position,
bcount);
} }
if (pc->bh)
iobuf(drive, pc, bcount);
else
xferfunc(drive, pc->current_position, bcount);
/* Update the current position */ /* Update the current position */
pc->actually_transferred += bcount; pc->actually_transferred += bcount;
pc->current_position += bcount; pc->current_position += bcount;
#if IDETAPE_DEBUG_LOG
if (tape->debug_level >= 2) debug_log(DBG_SENSE, "[cmd %x] transferred %d bytes on that intr.\n",
printk(KERN_INFO "ide-tape: [cmd %x] transferred %d bytes " pc->c[0], bcount);
"on that interrupt\n", pc->c[0], bcount);
#endif
/* And set the interrupt handler again */ /* And set the interrupt handler again */
ide_set_handler(drive, &idetape_pc_intr, IDETAPE_WAIT_CMD, NULL); ide_set_handler(drive, &idetape_pc_intr, IDETAPE_WAIT_CMD, NULL);
return ide_started; return ide_started;
} }
/* /*
* Packet Command Interface * Packet Command Interface
*
* The current Packet Command is available in tape->pc, and will not
* change until we finish handling it. Each packet command is associated
* with a callback function that will be called when the command is
* finished.
* *
* The handling will be done in three stages: * The current Packet Command is available in tape->pc, and will not change
* until we finish handling it. Each packet command is associated with a
* callback function that will be called when the command is finished.
* *
* 1. idetape_issue_packet_command will send the packet command to the * The handling will be done in three stages:
* drive, and will set the interrupt handler to idetape_pc_intr.
* *
* 2. On each interrupt, idetape_pc_intr will be called. This step * 1. idetape_issue_pc will send the packet command to the drive, and will set
* will be repeated until the device signals us that no more * the interrupt handler to idetape_pc_intr.
* interrupts will be issued.
* *
* 3. ATAPI Tape media access commands have immediate status with a * 2. On each interrupt, idetape_pc_intr will be called. This step will be
* delayed process. In case of a successful initiation of a * repeated until the device signals us that no more interrupts will be issued.
* media access packet command, the DSC bit will be set when the
* actual execution of the command is finished.
* Since the tape drive will not issue an interrupt, we have to
* poll for this event. In this case, we define the request as
* "low priority request" by setting rq_status to
* IDETAPE_RQ_POSTPONED, set a timer to poll for DSC and exit
* the driver.
* *
* ide.c will then give higher priority to requests which * 3. ATAPI Tape media access commands have immediate status with a delayed
* originate from the other device, until will change rq_status * process. In case of a successful initiation of a media access packet command,
* to RQ_ACTIVE. * the DSC bit will be set when the actual execution of the command is finished.
* Since the tape drive will not issue an interrupt, we have to poll for this
* event. In this case, we define the request as "low priority request" by
* setting rq_status to IDETAPE_RQ_POSTPONED, set a timer to poll for DSC and
* exit the driver.
* *
* 4. When the packet command is finished, it will be checked for errors. * ide.c will then give higher priority to requests which originate from the
* other device, until will change rq_status to RQ_ACTIVE.
* *
* 5. In case an error was found, we queue a request sense packet * 4. When the packet command is finished, it will be checked for errors.
* command in front of the request queue and retry the operation
* up to IDETAPE_MAX_PC_RETRIES times.
* *
* 6. In case no error was found, or we decided to give up and not * 5. In case an error was found, we queue a request sense packet command in
* to retry again, the callback function will be called and then * front of the request queue and retry the operation up to
* we will handle the next request. * IDETAPE_MAX_PC_RETRIES times.
* *
* 6. In case no error was found, or we decided to give up and not to retry
* again, the callback function will be called and then we will handle the next
* request.
*/ */
static ide_startstop_t idetape_transfer_pc(ide_drive_t *drive) static ide_startstop_t idetape_transfer_pc(ide_drive_t *drive)
{ {
...@@ -1388,8 +1233,9 @@ static ide_startstop_t idetape_transfer_pc(ide_drive_t *drive) ...@@ -1388,8 +1233,9 @@ static ide_startstop_t idetape_transfer_pc(ide_drive_t *drive)
ide_startstop_t startstop; ide_startstop_t startstop;
u8 ireason; u8 ireason;
if (ide_wait_stat(&startstop,drive,DRQ_STAT,BUSY_STAT,WAIT_READY)) { if (ide_wait_stat(&startstop, drive, DRQ_STAT, BUSY_STAT, WAIT_READY)) {
printk(KERN_ERR "ide-tape: Strange, packet command initiated yet DRQ isn't asserted\n"); printk(KERN_ERR "ide-tape: Strange, packet command initiated "
"yet DRQ isn't asserted\n");
return startstop; return startstop;
} }
ireason = hwif->INB(IDE_IREASON_REG); ireason = hwif->INB(IDE_IREASON_REG);
...@@ -1422,7 +1268,7 @@ static ide_startstop_t idetape_transfer_pc(ide_drive_t *drive) ...@@ -1422,7 +1268,7 @@ static ide_startstop_t idetape_transfer_pc(ide_drive_t *drive)
return ide_started; return ide_started;
} }
static ide_startstop_t idetape_issue_packet_command (ide_drive_t *drive, idetape_pc_t *pc) static ide_startstop_t idetape_issue_pc(ide_drive_t *drive, idetape_pc_t *pc)
{ {
ide_hwif_t *hwif = drive->hwif; ide_hwif_t *hwif = drive->hwif;
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
...@@ -1443,9 +1289,9 @@ static ide_startstop_t idetape_issue_packet_command (ide_drive_t *drive, idetape ...@@ -1443,9 +1289,9 @@ static ide_startstop_t idetape_issue_packet_command (ide_drive_t *drive, idetape
if (pc->retries > IDETAPE_MAX_PC_RETRIES || if (pc->retries > IDETAPE_MAX_PC_RETRIES ||
test_bit(PC_ABORT, &pc->flags)) { test_bit(PC_ABORT, &pc->flags)) {
/* /*
* We will "abort" retrying a packet command in case * We will "abort" retrying a packet command in case legitimate
* a legitimate error code was received (crossing a * error code was received (crossing a filemark, or end of the
* filemark, or end of the media, for example). * media, for example).
*/ */
if (!test_bit(PC_ABORT, &pc->flags)) { if (!test_bit(PC_ABORT, &pc->flags)) {
if (!(pc->c[0] == TEST_UNIT_READY && if (!(pc->c[0] == TEST_UNIT_READY &&
...@@ -1464,10 +1310,7 @@ static ide_startstop_t idetape_issue_packet_command (ide_drive_t *drive, idetape ...@@ -1464,10 +1310,7 @@ static ide_startstop_t idetape_issue_packet_command (ide_drive_t *drive, idetape
tape->failed_pc = NULL; tape->failed_pc = NULL;
return pc->callback(drive); return pc->callback(drive);
} }
#if IDETAPE_DEBUG_LOG debug_log(DBG_SENSE, "Retry #%d, cmd = %02X\n", pc->retries, pc->c[0]);
if (tape->debug_level >= 2)
printk(KERN_INFO "ide-tape: Retry number - %d, cmd = %02X\n", pc->retries, pc->c[0]);
#endif /* IDETAPE_DEBUG_LOG */
pc->retries++; pc->retries++;
/* We haven't transferred any data yet */ /* We haven't transferred any data yet */
...@@ -1499,31 +1342,24 @@ static ide_startstop_t idetape_issue_packet_command (ide_drive_t *drive, idetape ...@@ -1499,31 +1342,24 @@ static ide_startstop_t idetape_issue_packet_command (ide_drive_t *drive, idetape
} }
} }
/* static ide_startstop_t idetape_pc_callback(ide_drive_t *drive)
* General packet command callback function.
*/
static ide_startstop_t idetape_pc_callback (ide_drive_t *drive)
{ {
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
#if IDETAPE_DEBUG_LOG debug_log(DBG_PROCS, "Enter %s\n", __func__);
if (tape->debug_level >= 4)
printk(KERN_INFO "ide-tape: Reached idetape_pc_callback\n");
#endif /* IDETAPE_DEBUG_LOG */
idetape_end_request(drive, tape->pc->error ? 0 : 1, 0); idetape_end_request(drive, tape->pc->error ? 0 : 1, 0);
return ide_stopped; return ide_stopped;
} }
/* /* A mode sense command is used to "sense" tape parameters. */
* A mode sense command is used to "sense" tape parameters. static void idetape_create_mode_sense_cmd(idetape_pc_t *pc, u8 page_code)
*/
static void idetape_create_mode_sense_cmd (idetape_pc_t *pc, u8 page_code)
{ {
idetape_init_pc(pc); idetape_init_pc(pc);
pc->c[0] = MODE_SENSE; pc->c[0] = MODE_SENSE;
if (page_code != IDETAPE_BLOCK_DESCRIPTOR) if (page_code != IDETAPE_BLOCK_DESCRIPTOR)
pc->c[1] = 8; /* DBD = 1 - Don't return block descriptors */ /* DBD = 1 - Don't return block descriptors */
pc->c[1] = 8;
pc->c[2] = page_code; pc->c[2] = page_code;
/* /*
* Changed pc->c[3] to 0 (255 will at best return unused info). * Changed pc->c[3] to 0 (255 will at best return unused info).
...@@ -1533,7 +1369,8 @@ static void idetape_create_mode_sense_cmd (idetape_pc_t *pc, u8 page_code) ...@@ -1533,7 +1369,8 @@ static void idetape_create_mode_sense_cmd (idetape_pc_t *pc, u8 page_code)
* and return an error when 255 is used. * and return an error when 255 is used.
*/ */
pc->c[3] = 0; pc->c[3] = 0;
pc->c[4] = 255; /* (We will just discard data in that case) */ /* We will just discard data in that case */
pc->c[4] = 255;
if (page_code == IDETAPE_BLOCK_DESCRIPTOR) if (page_code == IDETAPE_BLOCK_DESCRIPTOR)
pc->request_transfer = 12; pc->request_transfer = 12;
else if (page_code == IDETAPE_CAPABILITIES_PAGE) else if (page_code == IDETAPE_CAPABILITIES_PAGE)
...@@ -1543,62 +1380,77 @@ static void idetape_create_mode_sense_cmd (idetape_pc_t *pc, u8 page_code) ...@@ -1543,62 +1380,77 @@ static void idetape_create_mode_sense_cmd (idetape_pc_t *pc, u8 page_code)
pc->callback = &idetape_pc_callback; pc->callback = &idetape_pc_callback;
} }
static void calculate_speeds(ide_drive_t *drive) static void idetape_calculate_speeds(ide_drive_t *drive)
{ {
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
int full = 125, empty = 75;
if (time_after(jiffies, tape->controlled_pipeline_head_time + 120 * HZ)) { if (time_after(jiffies,
tape->controlled_previous_pipeline_head = tape->controlled_last_pipeline_head; tape->controlled_pipeline_head_time + 120 * HZ)) {
tape->controlled_previous_head_time = tape->controlled_pipeline_head_time; tape->controlled_previous_pipeline_head =
tape->controlled_last_pipeline_head;
tape->controlled_previous_head_time =
tape->controlled_pipeline_head_time;
tape->controlled_last_pipeline_head = tape->pipeline_head; tape->controlled_last_pipeline_head = tape->pipeline_head;
tape->controlled_pipeline_head_time = jiffies; tape->controlled_pipeline_head_time = jiffies;
} }
if (time_after(jiffies, tape->controlled_pipeline_head_time + 60 * HZ)) if (time_after(jiffies, tape->controlled_pipeline_head_time + 60 * HZ))
tape->controlled_pipeline_head_speed = (tape->pipeline_head - tape->controlled_last_pipeline_head) * 32 * HZ / (jiffies - tape->controlled_pipeline_head_time); tape->controlled_pipeline_head_speed = (tape->pipeline_head -
tape->controlled_last_pipeline_head) * 32 * HZ /
(jiffies - tape->controlled_pipeline_head_time);
else if (time_after(jiffies, tape->controlled_previous_head_time)) else if (time_after(jiffies, tape->controlled_previous_head_time))
tape->controlled_pipeline_head_speed = (tape->pipeline_head - tape->controlled_previous_pipeline_head) * 32 * HZ / (jiffies - tape->controlled_previous_head_time); tape->controlled_pipeline_head_speed = (tape->pipeline_head -
tape->controlled_previous_pipeline_head) * 32 *
HZ / (jiffies - tape->controlled_previous_head_time);
if (tape->nr_pending_stages < tape->max_stages /*- 1 */) { if (tape->nr_pending_stages < tape->max_stages/*- 1 */) {
/* -1 for read mode error recovery */ /* -1 for read mode error recovery */
if (time_after(jiffies, tape->uncontrolled_previous_head_time + 10 * HZ)) { if (time_after(jiffies, tape->uncontrolled_previous_head_time +
10 * HZ)) {
tape->uncontrolled_pipeline_head_time = jiffies; tape->uncontrolled_pipeline_head_time = jiffies;
tape->uncontrolled_pipeline_head_speed = (tape->pipeline_head - tape->uncontrolled_previous_pipeline_head) * 32 * HZ / (jiffies - tape->uncontrolled_previous_head_time); tape->uncontrolled_pipeline_head_speed =
(tape->pipeline_head -
tape->uncontrolled_previous_pipeline_head) *
32 * HZ / (jiffies -
tape->uncontrolled_previous_head_time);
} }
} else { } else {
tape->uncontrolled_previous_head_time = jiffies; tape->uncontrolled_previous_head_time = jiffies;
tape->uncontrolled_previous_pipeline_head = tape->pipeline_head; tape->uncontrolled_previous_pipeline_head = tape->pipeline_head;
if (time_after(jiffies, tape->uncontrolled_pipeline_head_time + 30 * HZ)) { if (time_after(jiffies, tape->uncontrolled_pipeline_head_time +
30 * HZ))
tape->uncontrolled_pipeline_head_time = jiffies; tape->uncontrolled_pipeline_head_time = jiffies;
}
} }
tape->pipeline_head_speed = max(tape->uncontrolled_pipeline_head_speed, tape->controlled_pipeline_head_speed); tape->pipeline_head_speed = max(tape->uncontrolled_pipeline_head_speed,
if (tape->speed_control == 0) { tape->controlled_pipeline_head_speed);
tape->max_insert_speed = 5000;
} else if (tape->speed_control == 1) { if (tape->speed_control == 1) {
if (tape->nr_pending_stages >= tape->max_stages / 2) if (tape->nr_pending_stages >= tape->max_stages / 2)
tape->max_insert_speed = tape->pipeline_head_speed + tape->max_insert_speed = tape->pipeline_head_speed +
(1100 - tape->pipeline_head_speed) * 2 * (tape->nr_pending_stages - tape->max_stages / 2) / tape->max_stages; (1100 - tape->pipeline_head_speed) * 2 *
(tape->nr_pending_stages - tape->max_stages / 2)
/ tape->max_stages;
else else
tape->max_insert_speed = 500 + tape->max_insert_speed = 500 +
(tape->pipeline_head_speed - 500) * 2 * tape->nr_pending_stages / tape->max_stages; (tape->pipeline_head_speed - 500) * 2 *
tape->nr_pending_stages / tape->max_stages;
if (tape->nr_pending_stages >= tape->max_stages * 99 / 100) if (tape->nr_pending_stages >= tape->max_stages * 99 / 100)
tape->max_insert_speed = 5000; tape->max_insert_speed = 5000;
} else if (tape->speed_control == 2) {
tape->max_insert_speed = tape->pipeline_head_speed * empty / 100 +
(tape->pipeline_head_speed * full / 100 - tape->pipeline_head_speed * empty / 100) * tape->nr_pending_stages / tape->max_stages;
} else } else
tape->max_insert_speed = tape->speed_control; tape->max_insert_speed = tape->speed_control;
tape->max_insert_speed = max(tape->max_insert_speed, 500); tape->max_insert_speed = max(tape->max_insert_speed, 500);
} }
static ide_startstop_t idetape_media_access_finished (ide_drive_t *drive) static ide_startstop_t idetape_media_access_finished(ide_drive_t *drive)
{ {
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
idetape_pc_t *pc = tape->pc; idetape_pc_t *pc = tape->pc;
u8 stat; u8 stat;
stat = drive->hwif->INB(IDE_STATUS_REG); stat = ide_read_status(drive);
if (stat & SEEK_STAT) { if (stat & SEEK_STAT) {
if (stat & ERR_STAT) { if (stat & ERR_STAT) {
/* Error detected */ /* Error detected */
...@@ -1618,14 +1470,14 @@ static ide_startstop_t idetape_media_access_finished (ide_drive_t *drive) ...@@ -1618,14 +1470,14 @@ static ide_startstop_t idetape_media_access_finished (ide_drive_t *drive)
return pc->callback(drive); return pc->callback(drive);
} }
static ide_startstop_t idetape_rw_callback (ide_drive_t *drive) static ide_startstop_t idetape_rw_callback(ide_drive_t *drive)
{ {
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
struct request *rq = HWGROUP(drive)->rq; struct request *rq = HWGROUP(drive)->rq;
int blocks = tape->pc->actually_transferred / tape->tape_block_size; int blocks = tape->pc->actually_transferred / tape->blk_size;
tape->avg_size += blocks * tape->tape_block_size; tape->avg_size += blocks * tape->blk_size;
tape->insert_size += blocks * tape->tape_block_size; tape->insert_size += blocks * tape->blk_size;
if (tape->insert_size > 1024 * 1024) if (tape->insert_size > 1024 * 1024)
tape->measure_insert_time = 1; tape->measure_insert_time = 1;
if (tape->measure_insert_time) { if (tape->measure_insert_time) {
...@@ -1634,19 +1486,17 @@ static ide_startstop_t idetape_rw_callback (ide_drive_t *drive) ...@@ -1634,19 +1486,17 @@ static ide_startstop_t idetape_rw_callback (ide_drive_t *drive)
tape->insert_size = 0; tape->insert_size = 0;
} }
if (time_after(jiffies, tape->insert_time)) if (time_after(jiffies, tape->insert_time))
tape->insert_speed = tape->insert_size / 1024 * HZ / (jiffies - tape->insert_time); tape->insert_speed = tape->insert_size / 1024 * HZ /
(jiffies - tape->insert_time);
if (time_after_eq(jiffies, tape->avg_time + HZ)) { if (time_after_eq(jiffies, tape->avg_time + HZ)) {
tape->avg_speed = tape->avg_size * HZ / (jiffies - tape->avg_time) / 1024; tape->avg_speed = tape->avg_size * HZ /
(jiffies - tape->avg_time) / 1024;
tape->avg_size = 0; tape->avg_size = 0;
tape->avg_time = jiffies; tape->avg_time = jiffies;
} }
debug_log(DBG_PROCS, "Enter %s\n", __func__);
#if IDETAPE_DEBUG_LOG tape->first_frame += blocks;
if (tape->debug_level >= 4)
printk(KERN_INFO "ide-tape: Reached idetape_rw_callback\n");
#endif /* IDETAPE_DEBUG_LOG */
tape->first_frame_position += blocks;
rq->current_nr_sectors -= blocks; rq->current_nr_sectors -= blocks;
if (!tape->pc->error) if (!tape->pc->error)
...@@ -1656,7 +1506,8 @@ static ide_startstop_t idetape_rw_callback (ide_drive_t *drive) ...@@ -1656,7 +1506,8 @@ static ide_startstop_t idetape_rw_callback (ide_drive_t *drive)
return ide_stopped; return ide_stopped;
} }
static void idetape_create_read_cmd(idetape_tape_t *tape, idetape_pc_t *pc, unsigned int length, struct idetape_bh *bh) static void idetape_create_read_cmd(idetape_tape_t *tape, idetape_pc_t *pc,
unsigned int length, struct idetape_bh *bh)
{ {
idetape_init_pc(pc); idetape_init_pc(pc);
pc->c[0] = READ_6; pc->c[0] = READ_6;
...@@ -1666,12 +1517,14 @@ static void idetape_create_read_cmd(idetape_tape_t *tape, idetape_pc_t *pc, unsi ...@@ -1666,12 +1517,14 @@ static void idetape_create_read_cmd(idetape_tape_t *tape, idetape_pc_t *pc, unsi
pc->bh = bh; pc->bh = bh;
atomic_set(&bh->b_count, 0); atomic_set(&bh->b_count, 0);
pc->buffer = NULL; pc->buffer = NULL;
pc->request_transfer = pc->buffer_size = length * tape->tape_block_size; pc->buffer_size = length * tape->blk_size;
pc->request_transfer = pc->buffer_size;
if (pc->request_transfer == tape->stage_size) if (pc->request_transfer == tape->stage_size)
set_bit(PC_DMA_RECOMMENDED, &pc->flags); set_bit(PC_DMA_RECOMMENDED, &pc->flags);
} }
static void idetape_create_read_buffer_cmd(idetape_tape_t *tape, idetape_pc_t *pc, unsigned int length, struct idetape_bh *bh) static void idetape_create_read_buffer_cmd(idetape_tape_t *tape,
idetape_pc_t *pc, struct idetape_bh *bh)
{ {
int size = 32768; int size = 32768;
struct idetape_bh *p = bh; struct idetape_bh *p = bh;
...@@ -1689,10 +1542,12 @@ static void idetape_create_read_buffer_cmd(idetape_tape_t *tape, idetape_pc_t *p ...@@ -1689,10 +1542,12 @@ static void idetape_create_read_buffer_cmd(idetape_tape_t *tape, idetape_pc_t *p
atomic_set(&p->b_count, 0); atomic_set(&p->b_count, 0);
p = p->b_reqnext; p = p->b_reqnext;
} }
pc->request_transfer = pc->buffer_size = size; pc->request_transfer = size;
pc->buffer_size = size;
} }
static void idetape_create_write_cmd(idetape_tape_t *tape, idetape_pc_t *pc, unsigned int length, struct idetape_bh *bh) static void idetape_create_write_cmd(idetape_tape_t *tape, idetape_pc_t *pc,
unsigned int length, struct idetape_bh *bh)
{ {
idetape_init_pc(pc); idetape_init_pc(pc);
pc->c[0] = WRITE_6; pc->c[0] = WRITE_6;
...@@ -1704,14 +1559,12 @@ static void idetape_create_write_cmd(idetape_tape_t *tape, idetape_pc_t *pc, uns ...@@ -1704,14 +1559,12 @@ static void idetape_create_write_cmd(idetape_tape_t *tape, idetape_pc_t *pc, uns
pc->b_data = bh->b_data; pc->b_data = bh->b_data;
pc->b_count = atomic_read(&bh->b_count); pc->b_count = atomic_read(&bh->b_count);
pc->buffer = NULL; pc->buffer = NULL;
pc->request_transfer = pc->buffer_size = length * tape->tape_block_size; pc->buffer_size = length * tape->blk_size;
pc->request_transfer = pc->buffer_size;
if (pc->request_transfer == tape->stage_size) if (pc->request_transfer == tape->stage_size)
set_bit(PC_DMA_RECOMMENDED, &pc->flags); set_bit(PC_DMA_RECOMMENDED, &pc->flags);
} }
/*
* idetape_do_request is our request handling function.
*/
static ide_startstop_t idetape_do_request(ide_drive_t *drive, static ide_startstop_t idetape_do_request(ide_drive_t *drive,
struct request *rq, sector_t block) struct request *rq, sector_t block)
{ {
...@@ -1720,30 +1573,22 @@ static ide_startstop_t idetape_do_request(ide_drive_t *drive, ...@@ -1720,30 +1573,22 @@ static ide_startstop_t idetape_do_request(ide_drive_t *drive,
struct request *postponed_rq = tape->postponed_rq; struct request *postponed_rq = tape->postponed_rq;
u8 stat; u8 stat;
#if IDETAPE_DEBUG_LOG debug_log(DBG_SENSE, "sector: %ld, nr_sectors: %ld,"
if (tape->debug_level >= 2) " current_nr_sectors: %d\n",
printk(KERN_INFO "ide-tape: sector: %ld, "
"nr_sectors: %ld, current_nr_sectors: %d\n",
rq->sector, rq->nr_sectors, rq->current_nr_sectors); rq->sector, rq->nr_sectors, rq->current_nr_sectors);
#endif /* IDETAPE_DEBUG_LOG */
if (!blk_special_request(rq)) { if (!blk_special_request(rq)) {
/* /* We do not support buffer cache originated requests. */
* We do not support buffer cache originated requests.
*/
printk(KERN_NOTICE "ide-tape: %s: Unsupported request in " printk(KERN_NOTICE "ide-tape: %s: Unsupported request in "
"request queue (%d)\n", drive->name, rq->cmd_type); "request queue (%d)\n", drive->name, rq->cmd_type);
ide_end_request(drive, 0, 0); ide_end_request(drive, 0, 0);
return ide_stopped; return ide_stopped;
} }
/* /* Retry a failed packet command */
* Retry a failed packet command if (tape->failed_pc && tape->pc->c[0] == REQUEST_SENSE)
*/ return idetape_issue_pc(drive, tape->failed_pc);
if (tape->failed_pc != NULL &&
tape->pc->c[0] == REQUEST_SENSE) {
return idetape_issue_packet_command(drive, tape->failed_pc);
}
if (postponed_rq != NULL) if (postponed_rq != NULL)
if (rq != postponed_rq) { if (rq != postponed_rq) {
printk(KERN_ERR "ide-tape: ide-tape.c bug - " printk(KERN_ERR "ide-tape: ide-tape.c bug - "
...@@ -1758,7 +1603,7 @@ static ide_startstop_t idetape_do_request(ide_drive_t *drive, ...@@ -1758,7 +1603,7 @@ static ide_startstop_t idetape_do_request(ide_drive_t *drive,
* If the tape is still busy, postpone our request and service * If the tape is still busy, postpone our request and service
* the other device meanwhile. * the other device meanwhile.
*/ */
stat = drive->hwif->INB(IDE_STATUS_REG); stat = ide_read_status(drive);
if (!drive->dsc_overlap && !(rq->cmd[0] & REQ_IDETAPE_PC2)) if (!drive->dsc_overlap && !(rq->cmd[0] & REQ_IDETAPE_PC2))
set_bit(IDETAPE_IGNORE_DSC, &tape->flags); set_bit(IDETAPE_IGNORE_DSC, &tape->flags);
...@@ -1768,16 +1613,15 @@ static ide_startstop_t idetape_do_request(ide_drive_t *drive, ...@@ -1768,16 +1613,15 @@ static ide_startstop_t idetape_do_request(ide_drive_t *drive,
drive->post_reset = 0; drive->post_reset = 0;
} }
if (tape->tape_still_time > 100 && tape->tape_still_time < 200)
tape->measure_insert_time = 1;
if (time_after(jiffies, tape->insert_time)) if (time_after(jiffies, tape->insert_time))
tape->insert_speed = tape->insert_size / 1024 * HZ / (jiffies - tape->insert_time); tape->insert_speed = tape->insert_size / 1024 * HZ /
calculate_speeds(drive); (jiffies - tape->insert_time);
idetape_calculate_speeds(drive);
if (!test_and_clear_bit(IDETAPE_IGNORE_DSC, &tape->flags) && if (!test_and_clear_bit(IDETAPE_IGNORE_DSC, &tape->flags) &&
(stat & SEEK_STAT) == 0) { (stat & SEEK_STAT) == 0) {
if (postponed_rq == NULL) { if (postponed_rq == NULL) {
tape->dsc_polling_start = jiffies; tape->dsc_polling_start = jiffies;
tape->dsc_polling_frequency = tape->best_dsc_rw_frequency; tape->dsc_poll_freq = tape->best_dsc_rw_freq;
tape->dsc_timeout = jiffies + IDETAPE_DSC_RW_TIMEOUT; tape->dsc_timeout = jiffies + IDETAPE_DSC_RW_TIMEOUT;
} else if (time_after(jiffies, tape->dsc_timeout)) { } else if (time_after(jiffies, tape->dsc_timeout)) {
printk(KERN_ERR "ide-tape: %s: DSC timeout\n", printk(KERN_ERR "ide-tape: %s: DSC timeout\n",
...@@ -1788,8 +1632,10 @@ static ide_startstop_t idetape_do_request(ide_drive_t *drive, ...@@ -1788,8 +1632,10 @@ static ide_startstop_t idetape_do_request(ide_drive_t *drive,
} else { } else {
return ide_do_reset(drive); return ide_do_reset(drive);
} }
} else if (time_after(jiffies, tape->dsc_polling_start + IDETAPE_DSC_MA_THRESHOLD)) } else if (time_after(jiffies,
tape->dsc_polling_frequency = IDETAPE_DSC_MA_SLOW; tape->dsc_polling_start +
IDETAPE_DSC_MA_THRESHOLD))
tape->dsc_poll_freq = IDETAPE_DSC_MA_SLOW;
idetape_postpone_request(drive); idetape_postpone_request(drive);
return ide_stopped; return ide_stopped;
} }
...@@ -1797,20 +1643,23 @@ static ide_startstop_t idetape_do_request(ide_drive_t *drive, ...@@ -1797,20 +1643,23 @@ static ide_startstop_t idetape_do_request(ide_drive_t *drive,
tape->buffer_head++; tape->buffer_head++;
tape->postpone_cnt = 0; tape->postpone_cnt = 0;
pc = idetape_next_pc_storage(drive); pc = idetape_next_pc_storage(drive);
idetape_create_read_cmd(tape, pc, rq->current_nr_sectors, (struct idetape_bh *)rq->special); idetape_create_read_cmd(tape, pc, rq->current_nr_sectors,
(struct idetape_bh *)rq->special);
goto out; goto out;
} }
if (rq->cmd[0] & REQ_IDETAPE_WRITE) { if (rq->cmd[0] & REQ_IDETAPE_WRITE) {
tape->buffer_head++; tape->buffer_head++;
tape->postpone_cnt = 0; tape->postpone_cnt = 0;
pc = idetape_next_pc_storage(drive); pc = idetape_next_pc_storage(drive);
idetape_create_write_cmd(tape, pc, rq->current_nr_sectors, (struct idetape_bh *)rq->special); idetape_create_write_cmd(tape, pc, rq->current_nr_sectors,
(struct idetape_bh *)rq->special);
goto out; goto out;
} }
if (rq->cmd[0] & REQ_IDETAPE_READ_BUFFER) { if (rq->cmd[0] & REQ_IDETAPE_READ_BUFFER) {
tape->postpone_cnt = 0; tape->postpone_cnt = 0;
pc = idetape_next_pc_storage(drive); pc = idetape_next_pc_storage(drive);
idetape_create_read_buffer_cmd(tape, pc, rq->current_nr_sectors, (struct idetape_bh *)rq->special); idetape_create_read_buffer_cmd(tape, pc,
(struct idetape_bh *)rq->special);
goto out; goto out;
} }
if (rq->cmd[0] & REQ_IDETAPE_PC1) { if (rq->cmd[0] & REQ_IDETAPE_PC1) {
...@@ -1825,49 +1674,51 @@ static ide_startstop_t idetape_do_request(ide_drive_t *drive, ...@@ -1825,49 +1674,51 @@ static ide_startstop_t idetape_do_request(ide_drive_t *drive,
} }
BUG(); BUG();
out: out:
return idetape_issue_packet_command(drive, pc); return idetape_issue_pc(drive, pc);
} }
/* /* Pipeline related functions */
* Pipeline related functions static inline int idetape_pipeline_active(idetape_tape_t *tape)
*/
static inline int idetape_pipeline_active (idetape_tape_t *tape)
{ {
int rc1, rc2; int rc1, rc2;
rc1 = test_bit(IDETAPE_PIPELINE_ACTIVE, &tape->flags); rc1 = test_bit(IDETAPE_PIPELINE_ACTIVE, &tape->flags);
rc2 = (tape->active_data_request != NULL); rc2 = (tape->active_data_rq != NULL);
return rc1; return rc1;
} }
/* /*
* idetape_kmalloc_stage uses __get_free_page to allocate a pipeline * The function below uses __get_free_page to allocate a pipeline stage, along
* stage, along with all the necessary small buffers which together make * with all the necessary small buffers which together make a buffer of size
* a buffer of size tape->stage_size (or a bit more). We attempt to * tape->stage_size (or a bit more). We attempt to combine sequential pages as
* combine sequential pages as much as possible. * much as possible.
* *
* Returns a pointer to the new allocated stage, or NULL if we * It returns a pointer to the new allocated stage, or NULL if we can't (or
* can't (or don't want to) allocate a stage. * don't want to) allocate a stage.
* *
* Pipeline stages are optional and are used to increase performance. * Pipeline stages are optional and are used to increase performance. If we
* If we can't allocate them, we'll manage without them. * can't allocate them, we'll manage without them.
*/ */
static idetape_stage_t *__idetape_kmalloc_stage (idetape_tape_t *tape, int full, int clear) static idetape_stage_t *__idetape_kmalloc_stage(idetape_tape_t *tape, int full,
int clear)
{ {
idetape_stage_t *stage; idetape_stage_t *stage;
struct idetape_bh *prev_bh, *bh; struct idetape_bh *prev_bh, *bh;
int pages = tape->pages_per_stage; int pages = tape->pages_per_stage;
char *b_data = NULL; char *b_data = NULL;
if ((stage = kmalloc(sizeof (idetape_stage_t),GFP_KERNEL)) == NULL) stage = kmalloc(sizeof(idetape_stage_t), GFP_KERNEL);
if (!stage)
return NULL; return NULL;
stage->next = NULL; stage->next = NULL;
bh = stage->bh = kmalloc(sizeof(struct idetape_bh), GFP_KERNEL); stage->bh = kmalloc(sizeof(struct idetape_bh), GFP_KERNEL);
bh = stage->bh;
if (bh == NULL) if (bh == NULL)
goto abort; goto abort;
bh->b_reqnext = NULL; bh->b_reqnext = NULL;
if ((bh->b_data = (char *) __get_free_page (GFP_KERNEL)) == NULL) bh->b_data = (char *) __get_free_page(GFP_KERNEL);
if (!bh->b_data)
goto abort; goto abort;
if (clear) if (clear)
memset(bh->b_data, 0, PAGE_SIZE); memset(bh->b_data, 0, PAGE_SIZE);
...@@ -1875,7 +1726,8 @@ static idetape_stage_t *__idetape_kmalloc_stage (idetape_tape_t *tape, int full, ...@@ -1875,7 +1726,8 @@ static idetape_stage_t *__idetape_kmalloc_stage (idetape_tape_t *tape, int full,
atomic_set(&bh->b_count, full ? bh->b_size : 0); atomic_set(&bh->b_count, full ? bh->b_size : 0);
while (--pages) { while (--pages) {
if ((b_data = (char *) __get_free_page (GFP_KERNEL)) == NULL) b_data = (char *) __get_free_page(GFP_KERNEL);
if (!b_data)
goto abort; goto abort;
if (clear) if (clear)
memset(b_data, 0, PAGE_SIZE); memset(b_data, 0, PAGE_SIZE);
...@@ -1893,7 +1745,8 @@ static idetape_stage_t *__idetape_kmalloc_stage (idetape_tape_t *tape, int full, ...@@ -1893,7 +1745,8 @@ static idetape_stage_t *__idetape_kmalloc_stage (idetape_tape_t *tape, int full,
continue; continue;
} }
prev_bh = bh; prev_bh = bh;
if ((bh = kmalloc(sizeof(struct idetape_bh), GFP_KERNEL)) == NULL) { bh = kmalloc(sizeof(struct idetape_bh), GFP_KERNEL);
if (!bh) {
free_page((unsigned long) b_data); free_page((unsigned long) b_data);
goto abort; goto abort;
} }
...@@ -1912,14 +1765,11 @@ static idetape_stage_t *__idetape_kmalloc_stage (idetape_tape_t *tape, int full, ...@@ -1912,14 +1765,11 @@ static idetape_stage_t *__idetape_kmalloc_stage (idetape_tape_t *tape, int full,
return NULL; return NULL;
} }
static idetape_stage_t *idetape_kmalloc_stage (idetape_tape_t *tape) static idetape_stage_t *idetape_kmalloc_stage(idetape_tape_t *tape)
{ {
idetape_stage_t *cache_stage = tape->cache_stage; idetape_stage_t *cache_stage = tape->cache_stage;
#if IDETAPE_DEBUG_LOG debug_log(DBG_PROCS, "Enter %s\n", __func__);
if (tape->debug_level >= 4)
printk(KERN_INFO "ide-tape: Reached idetape_kmalloc_stage\n");
#endif /* IDETAPE_DEBUG_LOG */
if (tape->nr_stages >= tape->max_stages) if (tape->nr_stages >= tape->max_stages)
return NULL; return NULL;
...@@ -1930,7 +1780,8 @@ static idetape_stage_t *idetape_kmalloc_stage (idetape_tape_t *tape) ...@@ -1930,7 +1780,8 @@ static idetape_stage_t *idetape_kmalloc_stage (idetape_tape_t *tape)
return __idetape_kmalloc_stage(tape, 0, 0); return __idetape_kmalloc_stage(tape, 0, 0);
} }
static int idetape_copy_stage_from_user (idetape_tape_t *tape, idetape_stage_t *stage, const char __user *buf, int n) static int idetape_copy_stage_from_user(idetape_tape_t *tape,
idetape_stage_t *stage, const char __user *buf, int n)
{ {
struct idetape_bh *bh = tape->bh; struct idetape_bh *bh = tape->bh;
int count; int count;
...@@ -1938,12 +1789,15 @@ static int idetape_copy_stage_from_user (idetape_tape_t *tape, idetape_stage_t * ...@@ -1938,12 +1789,15 @@ static int idetape_copy_stage_from_user (idetape_tape_t *tape, idetape_stage_t *
while (n) { while (n) {
if (bh == NULL) { if (bh == NULL) {
printk(KERN_ERR "ide-tape: bh == NULL in " printk(KERN_ERR "ide-tape: bh == NULL in %s\n",
"idetape_copy_stage_from_user\n"); __func__);
return 1; return 1;
} }
count = min((unsigned int)(bh->b_size - atomic_read(&bh->b_count)), (unsigned int)n); count = min((unsigned int)
if (copy_from_user(bh->b_data + atomic_read(&bh->b_count), buf, count)) (bh->b_size - atomic_read(&bh->b_count)),
(unsigned int)n);
if (copy_from_user(bh->b_data + atomic_read(&bh->b_count), buf,
count))
ret = 1; ret = 1;
n -= count; n -= count;
atomic_add(count, &bh->b_count); atomic_add(count, &bh->b_count);
...@@ -1958,7 +1812,8 @@ static int idetape_copy_stage_from_user (idetape_tape_t *tape, idetape_stage_t * ...@@ -1958,7 +1812,8 @@ static int idetape_copy_stage_from_user (idetape_tape_t *tape, idetape_stage_t *
return ret; return ret;
} }
static int idetape_copy_stage_to_user (idetape_tape_t *tape, char __user *buf, idetape_stage_t *stage, int n) static int idetape_copy_stage_to_user(idetape_tape_t *tape, char __user *buf,
idetape_stage_t *stage, int n)
{ {
struct idetape_bh *bh = tape->bh; struct idetape_bh *bh = tape->bh;
int count; int count;
...@@ -1966,8 +1821,8 @@ static int idetape_copy_stage_to_user (idetape_tape_t *tape, char __user *buf, i ...@@ -1966,8 +1821,8 @@ static int idetape_copy_stage_to_user (idetape_tape_t *tape, char __user *buf, i
while (n) { while (n) {
if (bh == NULL) { if (bh == NULL) {
printk(KERN_ERR "ide-tape: bh == NULL in " printk(KERN_ERR "ide-tape: bh == NULL in %s\n",
"idetape_copy_stage_to_user\n"); __func__);
return 1; return 1;
} }
count = min(tape->b_count, n); count = min(tape->b_count, n);
...@@ -1978,7 +1833,8 @@ static int idetape_copy_stage_to_user (idetape_tape_t *tape, char __user *buf, i ...@@ -1978,7 +1833,8 @@ static int idetape_copy_stage_to_user (idetape_tape_t *tape, char __user *buf, i
tape->b_count -= count; tape->b_count -= count;
buf += count; buf += count;
if (!tape->b_count) { if (!tape->b_count) {
tape->bh = bh = bh->b_reqnext; bh = bh->b_reqnext;
tape->bh = bh;
if (bh) { if (bh) {
tape->b_data = bh->b_data; tape->b_data = bh->b_data;
tape->b_count = atomic_read(&bh->b_count); tape->b_count = atomic_read(&bh->b_count);
...@@ -1988,12 +1844,12 @@ static int idetape_copy_stage_to_user (idetape_tape_t *tape, char __user *buf, i ...@@ -1988,12 +1844,12 @@ static int idetape_copy_stage_to_user (idetape_tape_t *tape, char __user *buf, i
return ret; return ret;
} }
static void idetape_init_merge_stage (idetape_tape_t *tape) static void idetape_init_merge_stage(idetape_tape_t *tape)
{ {
struct idetape_bh *bh = tape->merge_stage->bh; struct idetape_bh *bh = tape->merge_stage->bh;
tape->bh = bh; tape->bh = bh;
if (tape->chrdev_direction == idetape_direction_write) if (tape->chrdev_dir == IDETAPE_DIR_WRITE)
atomic_set(&bh->b_count, 0); atomic_set(&bh->b_count, 0);
else { else {
tape->b_data = bh->b_data; tape->b_data = bh->b_data;
...@@ -2001,7 +1857,7 @@ static void idetape_init_merge_stage (idetape_tape_t *tape) ...@@ -2001,7 +1857,7 @@ static void idetape_init_merge_stage (idetape_tape_t *tape)
} }
} }
static void idetape_switch_buffers (idetape_tape_t *tape, idetape_stage_t *stage) static void idetape_switch_buffers(idetape_tape_t *tape, idetape_stage_t *stage)
{ {
struct idetape_bh *tmp; struct idetape_bh *tmp;
...@@ -2011,87 +1867,76 @@ static void idetape_switch_buffers (idetape_tape_t *tape, idetape_stage_t *stage ...@@ -2011,87 +1867,76 @@ static void idetape_switch_buffers (idetape_tape_t *tape, idetape_stage_t *stage
idetape_init_merge_stage(tape); idetape_init_merge_stage(tape);
} }
/* /* Add a new stage at the end of the pipeline. */
* idetape_add_stage_tail adds a new stage at the end of the pipeline. static void idetape_add_stage_tail(ide_drive_t *drive, idetape_stage_t *stage)
*/
static void idetape_add_stage_tail (ide_drive_t *drive,idetape_stage_t *stage)
{ {
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
unsigned long flags; unsigned long flags;
#if IDETAPE_DEBUG_LOG debug_log(DBG_PROCS, "Enter %s\n", __func__);
if (tape->debug_level >= 4)
printk (KERN_INFO "ide-tape: Reached idetape_add_stage_tail\n"); spin_lock_irqsave(&tape->lock, flags);
#endif /* IDETAPE_DEBUG_LOG */
spin_lock_irqsave(&tape->spinlock, flags);
stage->next = NULL; stage->next = NULL;
if (tape->last_stage != NULL) if (tape->last_stage != NULL)
tape->last_stage->next=stage; tape->last_stage->next = stage;
else else
tape->first_stage = tape->next_stage=stage; tape->first_stage = stage;
tape->next_stage = stage;
tape->last_stage = stage; tape->last_stage = stage;
if (tape->next_stage == NULL) if (tape->next_stage == NULL)
tape->next_stage = tape->last_stage; tape->next_stage = tape->last_stage;
tape->nr_stages++; tape->nr_stages++;
tape->nr_pending_stages++; tape->nr_pending_stages++;
spin_unlock_irqrestore(&tape->spinlock, flags); spin_unlock_irqrestore(&tape->lock, flags);
} }
/* /* Install a completion in a pending request and sleep until it is serviced. The
* idetape_wait_for_request installs a completion in a pending request * caller should ensure that the request will not be serviced before we install
* and sleeps until it is serviced. * the completion (usually by disabling interrupts).
*
* The caller should ensure that the request will not be serviced
* before we install the completion (usually by disabling interrupts).
*/ */
static void idetape_wait_for_request (ide_drive_t *drive, struct request *rq) static void idetape_wait_for_request(ide_drive_t *drive, struct request *rq)
{ {
DECLARE_COMPLETION_ONSTACK(wait); DECLARE_COMPLETION_ONSTACK(wait);
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
if (rq == NULL || !blk_special_request(rq)) { if (rq == NULL || !blk_special_request(rq)) {
printk (KERN_ERR "ide-tape: bug: Trying to sleep on non-valid request\n"); printk(KERN_ERR "ide-tape: bug: Trying to sleep on non-valid"
" request\n");
return; return;
} }
rq->end_io_data = &wait; rq->end_io_data = &wait;
rq->end_io = blk_end_sync_rq; rq->end_io = blk_end_sync_rq;
spin_unlock_irq(&tape->spinlock); spin_unlock_irq(&tape->lock);
wait_for_completion(&wait); wait_for_completion(&wait);
/* The stage and its struct request have been deallocated */ /* The stage and its struct request have been deallocated */
spin_lock_irq(&tape->spinlock); spin_lock_irq(&tape->lock);
} }
static ide_startstop_t idetape_read_position_callback (ide_drive_t *drive) static ide_startstop_t idetape_read_position_callback(ide_drive_t *drive)
{ {
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
idetape_read_position_result_t *result; u8 *readpos = tape->pc->buffer;
#if IDETAPE_DEBUG_LOG debug_log(DBG_PROCS, "Enter %s\n", __func__);
if (tape->debug_level >= 4)
printk(KERN_INFO "ide-tape: Reached idetape_read_position_callback\n");
#endif /* IDETAPE_DEBUG_LOG */
if (!tape->pc->error) { if (!tape->pc->error) {
result = (idetape_read_position_result_t *) tape->pc->buffer; debug_log(DBG_SENSE, "BOP - %s\n",
#if IDETAPE_DEBUG_LOG (readpos[0] & 0x80) ? "Yes" : "No");
if (tape->debug_level >= 2) debug_log(DBG_SENSE, "EOP - %s\n",
printk(KERN_INFO "ide-tape: BOP - %s\n",result->bop ? "Yes":"No"); (readpos[0] & 0x40) ? "Yes" : "No");
if (tape->debug_level >= 2)
printk(KERN_INFO "ide-tape: EOP - %s\n",result->eop ? "Yes":"No"); if (readpos[0] & 0x4) {
#endif /* IDETAPE_DEBUG_LOG */ printk(KERN_INFO "ide-tape: Block location is unknown"
if (result->bpu) { "to the tape\n");
printk(KERN_INFO "ide-tape: Block location is unknown to the tape\n");
clear_bit(IDETAPE_ADDRESS_VALID, &tape->flags); clear_bit(IDETAPE_ADDRESS_VALID, &tape->flags);
idetape_end_request(drive, 0, 0); idetape_end_request(drive, 0, 0);
} else { } else {
#if IDETAPE_DEBUG_LOG debug_log(DBG_SENSE, "Block Location - %u\n",
if (tape->debug_level >= 2) be32_to_cpu(*(u32 *)&readpos[4]));
printk(KERN_INFO "ide-tape: Block Location - %u\n", ntohl(result->first_block));
#endif /* IDETAPE_DEBUG_LOG */ tape->partition = readpos[1];
tape->partition = result->partition; tape->first_frame =
tape->first_frame_position = ntohl(result->first_block); be32_to_cpu(*(u32 *)&readpos[4]);
tape->last_frame_position = ntohl(result->last_block);
tape->blocks_in_buffer = result->blocks_in_buffer[2];
set_bit(IDETAPE_ADDRESS_VALID, &tape->flags); set_bit(IDETAPE_ADDRESS_VALID, &tape->flags);
idetape_end_request(drive, 1, 0); idetape_end_request(drive, 1, 0);
} }
...@@ -2102,14 +1947,11 @@ static ide_startstop_t idetape_read_position_callback (ide_drive_t *drive) ...@@ -2102,14 +1947,11 @@ static ide_startstop_t idetape_read_position_callback (ide_drive_t *drive)
} }
/* /*
* idetape_create_write_filemark_cmd will: * Write a filemark if write_filemark=1. Flush the device buffers without
* * writing a filemark otherwise.
* 1. Write a filemark if write_filemark=1.
* 2. Flush the device buffers without writing a filemark
* if write_filemark=0.
*
*/ */
static void idetape_create_write_filemark_cmd (ide_drive_t *drive, idetape_pc_t *pc,int write_filemark) static void idetape_create_write_filemark_cmd(ide_drive_t *drive,
idetape_pc_t *pc, int write_filemark)
{ {
idetape_init_pc(pc); idetape_init_pc(pc);
pc->c[0] = WRITE_FILEMARKS; pc->c[0] = WRITE_FILEMARKS;
...@@ -2126,26 +1968,19 @@ static void idetape_create_test_unit_ready_cmd(idetape_pc_t *pc) ...@@ -2126,26 +1968,19 @@ static void idetape_create_test_unit_ready_cmd(idetape_pc_t *pc)
} }
/* /*
* idetape_queue_pc_tail is based on the following functions: * We add a special packet command request to the tail of the request queue, and
* * wait for it to be serviced. This is not to be called from within the request
* ide_do_drive_cmd from ide.c * handling part of the driver! We allocate here data on the stack and it is
* cdrom_queue_request and cdrom_queue_packet_command from ide-cd.c * valid until the request is finished. This is not the case for the bottom part
* of the driver, where we are always leaving the functions to wait for an
* interrupt or a timer event.
* *
* We add a special packet command request to the tail of the request * From the bottom part of the driver, we should allocate safe memory using
* queue, and wait for it to be serviced. * idetape_next_pc_storage() and ide_tape_next_rq_storage(), and add the request
* * to the request list without waiting for it to be serviced! In that case, we
* This is not to be called from within the request handling part * usually use idetape_queue_pc_head().
* of the driver ! We allocate here data in the stack, and it is valid
* until the request is finished. This is not the case for the bottom
* part of the driver, where we are always leaving the functions to wait
* for an interrupt or a timer event.
*
* From the bottom part of the driver, we should allocate safe memory
* using idetape_next_pc_storage and idetape_next_rq_storage, and add
* the request to the request list without waiting for it to be serviced !
* In that case, we usually use idetape_queue_pc_head.
*/ */
static int __idetape_queue_pc_tail (ide_drive_t *drive, idetape_pc_t *pc) static int __idetape_queue_pc_tail(ide_drive_t *drive, idetape_pc_t *pc)
{ {
struct ide_tape_obj *tape = drive->driver_data; struct ide_tape_obj *tape = drive->driver_data;
struct request rq; struct request rq;
...@@ -2156,7 +1991,8 @@ static int __idetape_queue_pc_tail (ide_drive_t *drive, idetape_pc_t *pc) ...@@ -2156,7 +1991,8 @@ static int __idetape_queue_pc_tail (ide_drive_t *drive, idetape_pc_t *pc)
return ide_do_drive_cmd(drive, &rq, ide_wait); return ide_do_drive_cmd(drive, &rq, ide_wait);
} }
static void idetape_create_load_unload_cmd (ide_drive_t *drive, idetape_pc_t *pc,int cmd) static void idetape_create_load_unload_cmd(ide_drive_t *drive, idetape_pc_t *pc,
int cmd)
{ {
idetape_init_pc(pc); idetape_init_pc(pc);
pc->c[0] = START_STOP; pc->c[0] = START_STOP;
...@@ -2171,9 +2007,7 @@ static int idetape_wait_ready(ide_drive_t *drive, unsigned long timeout) ...@@ -2171,9 +2007,7 @@ static int idetape_wait_ready(ide_drive_t *drive, unsigned long timeout)
idetape_pc_t pc; idetape_pc_t pc;
int load_attempted = 0; int load_attempted = 0;
/* /* Wait for the tape to become ready */
* Wait for the tape to become ready
*/
set_bit(IDETAPE_MEDIUM_PRESENT, &tape->flags); set_bit(IDETAPE_MEDIUM_PRESENT, &tape->flags);
timeout += jiffies; timeout += jiffies;
while (time_before(jiffies, timeout)) { while (time_before(jiffies, timeout)) {
...@@ -2181,10 +2015,12 @@ static int idetape_wait_ready(ide_drive_t *drive, unsigned long timeout) ...@@ -2181,10 +2015,12 @@ static int idetape_wait_ready(ide_drive_t *drive, unsigned long timeout)
if (!__idetape_queue_pc_tail(drive, &pc)) if (!__idetape_queue_pc_tail(drive, &pc))
return 0; return 0;
if ((tape->sense_key == 2 && tape->asc == 4 && tape->ascq == 2) if ((tape->sense_key == 2 && tape->asc == 4 && tape->ascq == 2)
|| (tape->asc == 0x3A)) { /* no media */ || (tape->asc == 0x3A)) {
/* no media */
if (load_attempted) if (load_attempted)
return -ENOMEDIUM; return -ENOMEDIUM;
idetape_create_load_unload_cmd(drive, &pc, IDETAPE_LU_LOAD_MASK); idetape_create_load_unload_cmd(drive, &pc,
IDETAPE_LU_LOAD_MASK);
__idetape_queue_pc_tail(drive, &pc); __idetape_queue_pc_tail(drive, &pc);
load_attempted = 1; load_attempted = 1;
/* not about to be ready */ /* not about to be ready */
...@@ -2196,24 +2032,25 @@ static int idetape_wait_ready(ide_drive_t *drive, unsigned long timeout) ...@@ -2196,24 +2032,25 @@ static int idetape_wait_ready(ide_drive_t *drive, unsigned long timeout)
return -EIO; return -EIO;
} }
static int idetape_queue_pc_tail (ide_drive_t *drive,idetape_pc_t *pc) static int idetape_queue_pc_tail(ide_drive_t *drive, idetape_pc_t *pc)
{ {
return __idetape_queue_pc_tail(drive, pc); return __idetape_queue_pc_tail(drive, pc);
} }
static int idetape_flush_tape_buffers (ide_drive_t *drive) static int idetape_flush_tape_buffers(ide_drive_t *drive)
{ {
idetape_pc_t pc; idetape_pc_t pc;
int rc; int rc;
idetape_create_write_filemark_cmd(drive, &pc, 0); idetape_create_write_filemark_cmd(drive, &pc, 0);
if ((rc = idetape_queue_pc_tail(drive, &pc))) rc = idetape_queue_pc_tail(drive, &pc);
if (rc)
return rc; return rc;
idetape_wait_ready(drive, 60 * 5 * HZ); idetape_wait_ready(drive, 60 * 5 * HZ);
return 0; return 0;
} }
static void idetape_create_read_position_cmd (idetape_pc_t *pc) static void idetape_create_read_position_cmd(idetape_pc_t *pc)
{ {
idetape_init_pc(pc); idetape_init_pc(pc);
pc->c[0] = READ_POSITION; pc->c[0] = READ_POSITION;
...@@ -2221,25 +2058,23 @@ static void idetape_create_read_position_cmd (idetape_pc_t *pc) ...@@ -2221,25 +2058,23 @@ static void idetape_create_read_position_cmd (idetape_pc_t *pc)
pc->callback = &idetape_read_position_callback; pc->callback = &idetape_read_position_callback;
} }
static int idetape_read_position (ide_drive_t *drive) static int idetape_read_position(ide_drive_t *drive)
{ {
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
idetape_pc_t pc; idetape_pc_t pc;
int position; int position;
#if IDETAPE_DEBUG_LOG debug_log(DBG_PROCS, "Enter %s\n", __func__);
if (tape->debug_level >= 4)
printk(KERN_INFO "ide-tape: Reached idetape_read_position\n");
#endif /* IDETAPE_DEBUG_LOG */
idetape_create_read_position_cmd(&pc); idetape_create_read_position_cmd(&pc);
if (idetape_queue_pc_tail(drive, &pc)) if (idetape_queue_pc_tail(drive, &pc))
return -1; return -1;
position = tape->first_frame_position; position = tape->first_frame;
return position; return position;
} }
static void idetape_create_locate_cmd (ide_drive_t *drive, idetape_pc_t *pc, unsigned int block, u8 partition, int skip) static void idetape_create_locate_cmd(ide_drive_t *drive, idetape_pc_t *pc,
unsigned int block, u8 partition, int skip)
{ {
idetape_init_pc(pc); idetape_init_pc(pc);
pc->c[0] = POSITION_TO_ELEMENT; pc->c[0] = POSITION_TO_ELEMENT;
...@@ -2250,7 +2085,8 @@ static void idetape_create_locate_cmd (ide_drive_t *drive, idetape_pc_t *pc, uns ...@@ -2250,7 +2085,8 @@ static void idetape_create_locate_cmd (ide_drive_t *drive, idetape_pc_t *pc, uns
pc->callback = &idetape_pc_callback; pc->callback = &idetape_pc_callback;
} }
static int idetape_create_prevent_cmd (ide_drive_t *drive, idetape_pc_t *pc, int prevent) static int idetape_create_prevent_cmd(ide_drive_t *drive, idetape_pc_t *pc,
int prevent)
{ {
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
...@@ -2265,17 +2101,17 @@ static int idetape_create_prevent_cmd (ide_drive_t *drive, idetape_pc_t *pc, int ...@@ -2265,17 +2101,17 @@ static int idetape_create_prevent_cmd (ide_drive_t *drive, idetape_pc_t *pc, int
return 1; return 1;
} }
static int __idetape_discard_read_pipeline (ide_drive_t *drive) static int __idetape_discard_read_pipeline(ide_drive_t *drive)
{ {
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
unsigned long flags; unsigned long flags;
int cnt; int cnt;
if (tape->chrdev_direction != idetape_direction_read) if (tape->chrdev_dir != IDETAPE_DIR_READ)
return 0; return 0;
/* Remove merge stage. */ /* Remove merge stage. */
cnt = tape->merge_stage_size / tape->tape_block_size; cnt = tape->merge_stage_size / tape->blk_size;
if (test_and_clear_bit(IDETAPE_FILEMARK, &tape->flags)) if (test_and_clear_bit(IDETAPE_FILEMARK, &tape->flags))
++cnt; /* Filemarks count as 1 sector */ ++cnt; /* Filemarks count as 1 sector */
tape->merge_stage_size = 0; tape->merge_stage_size = 0;
...@@ -2286,22 +2122,22 @@ static int __idetape_discard_read_pipeline (ide_drive_t *drive) ...@@ -2286,22 +2122,22 @@ static int __idetape_discard_read_pipeline (ide_drive_t *drive)
/* Clear pipeline flags. */ /* Clear pipeline flags. */
clear_bit(IDETAPE_PIPELINE_ERROR, &tape->flags); clear_bit(IDETAPE_PIPELINE_ERROR, &tape->flags);
tape->chrdev_direction = idetape_direction_none; tape->chrdev_dir = IDETAPE_DIR_NONE;
/* Remove pipeline stages. */ /* Remove pipeline stages. */
if (tape->first_stage == NULL) if (tape->first_stage == NULL)
return 0; return 0;
spin_lock_irqsave(&tape->spinlock, flags); spin_lock_irqsave(&tape->lock, flags);
tape->next_stage = NULL; tape->next_stage = NULL;
if (idetape_pipeline_active(tape)) if (idetape_pipeline_active(tape))
idetape_wait_for_request(drive, tape->active_data_request); idetape_wait_for_request(drive, tape->active_data_rq);
spin_unlock_irqrestore(&tape->spinlock, flags); spin_unlock_irqrestore(&tape->lock, flags);
while (tape->first_stage != NULL) { while (tape->first_stage != NULL) {
struct request *rq_ptr = &tape->first_stage->rq; struct request *rq_ptr = &tape->first_stage->rq;
cnt += rq_ptr->nr_sectors - rq_ptr->current_nr_sectors; cnt += rq_ptr->nr_sectors - rq_ptr->current_nr_sectors;
if (rq_ptr->errors == IDETAPE_ERROR_FILEMARK) if (rq_ptr->errors == IDETAPE_ERROR_FILEMARK)
++cnt; ++cnt;
idetape_remove_stage_head(drive); idetape_remove_stage_head(drive);
...@@ -2312,21 +2148,19 @@ static int __idetape_discard_read_pipeline (ide_drive_t *drive) ...@@ -2312,21 +2148,19 @@ static int __idetape_discard_read_pipeline (ide_drive_t *drive)
} }
/* /*
* idetape_position_tape positions the tape to the requested block * Position the tape to the requested block using the LOCATE packet command.
* using the LOCATE packet command. A READ POSITION command is then * A READ POSITION command is then issued to check where we are positioned. Like
* issued to check where we are positioned. * all higher level operations, we queue the commands at the tail of the request
* * queue and wait for their completion.
* Like all higher level operations, we queue the commands at the tail
* of the request queue and wait for their completion.
*
*/ */
static int idetape_position_tape (ide_drive_t *drive, unsigned int block, u8 partition, int skip) static int idetape_position_tape(ide_drive_t *drive, unsigned int block,
u8 partition, int skip)
{ {
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
int retval; int retval;
idetape_pc_t pc; idetape_pc_t pc;
if (tape->chrdev_direction == idetape_direction_read) if (tape->chrdev_dir == IDETAPE_DIR_READ)
__idetape_discard_read_pipeline(drive); __idetape_discard_read_pipeline(drive);
idetape_wait_ready(drive, 60 * 5 * HZ); idetape_wait_ready(drive, 60 * 5 * HZ);
idetape_create_locate_cmd(drive, &pc, block, partition, skip); idetape_create_locate_cmd(drive, &pc, block, partition, skip);
...@@ -2338,7 +2172,8 @@ static int idetape_position_tape (ide_drive_t *drive, unsigned int block, u8 par ...@@ -2338,7 +2172,8 @@ static int idetape_position_tape (ide_drive_t *drive, unsigned int block, u8 par
return (idetape_queue_pc_tail(drive, &pc)); return (idetape_queue_pc_tail(drive, &pc));
} }
static void idetape_discard_read_pipeline (ide_drive_t *drive, int restore_position) static void idetape_discard_read_pipeline(ide_drive_t *drive,
int restore_position)
{ {
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
int cnt; int cnt;
...@@ -2349,35 +2184,37 @@ static void idetape_discard_read_pipeline (ide_drive_t *drive, int restore_posit ...@@ -2349,35 +2184,37 @@ static void idetape_discard_read_pipeline (ide_drive_t *drive, int restore_posit
position = idetape_read_position(drive); position = idetape_read_position(drive);
seek = position > cnt ? position - cnt : 0; seek = position > cnt ? position - cnt : 0;
if (idetape_position_tape(drive, seek, 0, 0)) { if (idetape_position_tape(drive, seek, 0, 0)) {
printk(KERN_INFO "ide-tape: %s: position_tape failed in discard_pipeline()\n", tape->name); printk(KERN_INFO "ide-tape: %s: position_tape failed in"
" discard_pipeline()\n", tape->name);
return; return;
} }
} }
} }
/* /*
* idetape_queue_rw_tail generates a read/write request for the block * Generate a read/write request for the block device interface and wait for it
* device interface and wait for it to be serviced. * to be serviced.
*/ */
static int idetape_queue_rw_tail(ide_drive_t *drive, int cmd, int blocks, struct idetape_bh *bh) static int idetape_queue_rw_tail(ide_drive_t *drive, int cmd, int blocks,
struct idetape_bh *bh)
{ {
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
struct request rq; struct request rq;
#if IDETAPE_DEBUG_LOG debug_log(DBG_SENSE, "%s: cmd=%d\n", __func__, cmd);
if (tape->debug_level >= 2)
printk(KERN_INFO "ide-tape: idetape_queue_rw_tail: cmd=%d\n",cmd);
#endif /* IDETAPE_DEBUG_LOG */
if (idetape_pipeline_active(tape)) { if (idetape_pipeline_active(tape)) {
printk(KERN_ERR "ide-tape: bug: the pipeline is active in idetape_queue_rw_tail\n"); printk(KERN_ERR "ide-tape: bug: the pipeline is active in %s\n",
__func__);
return (0); return (0);
} }
idetape_init_rq(&rq, cmd); idetape_init_rq(&rq, cmd);
rq.rq_disk = tape->disk; rq.rq_disk = tape->disk;
rq.special = (void *)bh; rq.special = (void *)bh;
rq.sector = tape->first_frame_position; rq.sector = tape->first_frame;
rq.nr_sectors = rq.current_nr_sectors = blocks; rq.nr_sectors = blocks;
rq.current_nr_sectors = blocks;
(void) ide_do_drive_cmd(drive, &rq, ide_wait); (void) ide_do_drive_cmd(drive, &rq, ide_wait);
if ((cmd & (REQ_IDETAPE_READ | REQ_IDETAPE_WRITE)) == 0) if ((cmd & (REQ_IDETAPE_READ | REQ_IDETAPE_WRITE)) == 0)
...@@ -2387,14 +2224,11 @@ static int idetape_queue_rw_tail(ide_drive_t *drive, int cmd, int blocks, struct ...@@ -2387,14 +2224,11 @@ static int idetape_queue_rw_tail(ide_drive_t *drive, int cmd, int blocks, struct
idetape_init_merge_stage(tape); idetape_init_merge_stage(tape);
if (rq.errors == IDETAPE_ERROR_GENERAL) if (rq.errors == IDETAPE_ERROR_GENERAL)
return -EIO; return -EIO;
return (tape->tape_block_size * (blocks-rq.current_nr_sectors)); return (tape->blk_size * (blocks-rq.current_nr_sectors));
} }
/* /* start servicing the pipeline stages, starting from tape->next_stage. */
* idetape_insert_pipeline_into_queue is used to start servicing the static void idetape_plug_pipeline(ide_drive_t *drive)
* pipeline stages, starting from tape->next_stage.
*/
static void idetape_insert_pipeline_into_queue (ide_drive_t *drive)
{ {
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
...@@ -2403,19 +2237,20 @@ static void idetape_insert_pipeline_into_queue (ide_drive_t *drive) ...@@ -2403,19 +2237,20 @@ static void idetape_insert_pipeline_into_queue (ide_drive_t *drive)
if (!idetape_pipeline_active(tape)) { if (!idetape_pipeline_active(tape)) {
set_bit(IDETAPE_PIPELINE_ACTIVE, &tape->flags); set_bit(IDETAPE_PIPELINE_ACTIVE, &tape->flags);
idetape_activate_next_stage(drive); idetape_activate_next_stage(drive);
(void) ide_do_drive_cmd(drive, tape->active_data_request, ide_end); (void) ide_do_drive_cmd(drive, tape->active_data_rq, ide_end);
} }
} }
static void idetape_create_inquiry_cmd (idetape_pc_t *pc) static void idetape_create_inquiry_cmd(idetape_pc_t *pc)
{ {
idetape_init_pc(pc); idetape_init_pc(pc);
pc->c[0] = INQUIRY; pc->c[0] = INQUIRY;
pc->c[4] = pc->request_transfer = 254; pc->c[4] = 254;
pc->request_transfer = 254;
pc->callback = &idetape_pc_callback; pc->callback = &idetape_pc_callback;
} }
static void idetape_create_rewind_cmd (ide_drive_t *drive, idetape_pc_t *pc) static void idetape_create_rewind_cmd(ide_drive_t *drive, idetape_pc_t *pc)
{ {
idetape_init_pc(pc); idetape_init_pc(pc);
pc->c[0] = REZERO_UNIT; pc->c[0] = REZERO_UNIT;
...@@ -2423,7 +2258,7 @@ static void idetape_create_rewind_cmd (ide_drive_t *drive, idetape_pc_t *pc) ...@@ -2423,7 +2258,7 @@ static void idetape_create_rewind_cmd (ide_drive_t *drive, idetape_pc_t *pc)
pc->callback = &idetape_pc_callback; pc->callback = &idetape_pc_callback;
} }
static void idetape_create_erase_cmd (idetape_pc_t *pc) static void idetape_create_erase_cmd(idetape_pc_t *pc)
{ {
idetape_init_pc(pc); idetape_init_pc(pc);
pc->c[0] = ERASE; pc->c[0] = ERASE;
...@@ -2432,7 +2267,7 @@ static void idetape_create_erase_cmd (idetape_pc_t *pc) ...@@ -2432,7 +2267,7 @@ static void idetape_create_erase_cmd (idetape_pc_t *pc)
pc->callback = &idetape_pc_callback; pc->callback = &idetape_pc_callback;
} }
static void idetape_create_space_cmd (idetape_pc_t *pc,int count, u8 cmd) static void idetape_create_space_cmd(idetape_pc_t *pc, int count, u8 cmd)
{ {
idetape_init_pc(pc); idetape_init_pc(pc);
pc->c[0] = SPACE; pc->c[0] = SPACE;
...@@ -2442,89 +2277,87 @@ static void idetape_create_space_cmd (idetape_pc_t *pc,int count, u8 cmd) ...@@ -2442,89 +2277,87 @@ static void idetape_create_space_cmd (idetape_pc_t *pc,int count, u8 cmd)
pc->callback = &idetape_pc_callback; pc->callback = &idetape_pc_callback;
} }
static void idetape_wait_first_stage (ide_drive_t *drive) static void idetape_wait_first_stage(ide_drive_t *drive)
{ {
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
unsigned long flags; unsigned long flags;
if (tape->first_stage == NULL) if (tape->first_stage == NULL)
return; return;
spin_lock_irqsave(&tape->spinlock, flags); spin_lock_irqsave(&tape->lock, flags);
if (tape->active_stage == tape->first_stage) if (tape->active_stage == tape->first_stage)
idetape_wait_for_request(drive, tape->active_data_request); idetape_wait_for_request(drive, tape->active_data_rq);
spin_unlock_irqrestore(&tape->spinlock, flags); spin_unlock_irqrestore(&tape->lock, flags);
} }
/* /*
* idetape_add_chrdev_write_request tries to add a character device * Try to add a character device originated write request to our pipeline. In
* originated write request to our pipeline. In case we don't succeed, * case we don't succeed, we revert to non-pipelined operation mode for this
* we revert to non-pipelined operation mode for this request. * request. In order to accomplish that, we
* *
* 1. Try to allocate a new pipeline stage. * 1. Try to allocate a new pipeline stage.
* 2. If we can't, wait for more and more requests to be serviced * 2. If we can't, wait for more and more requests to be serviced and try again
* and try again each time. * each time.
* 3. If we still can't allocate a stage, fallback to * 3. If we still can't allocate a stage, fallback to non-pipelined operation
* non-pipelined operation mode for this request. * mode for this request.
*/ */
static int idetape_add_chrdev_write_request (ide_drive_t *drive, int blocks) static int idetape_add_chrdev_write_request(ide_drive_t *drive, int blocks)
{ {
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
idetape_stage_t *new_stage; idetape_stage_t *new_stage;
unsigned long flags; unsigned long flags;
struct request *rq; struct request *rq;
#if IDETAPE_DEBUG_LOG debug_log(DBG_CHRDEV, "Enter %s\n", __func__);
if (tape->debug_level >= 3)
printk(KERN_INFO "ide-tape: Reached idetape_add_chrdev_write_request\n");
#endif /* IDETAPE_DEBUG_LOG */
/* /* Attempt to allocate a new stage. Beware possible race conditions. */
* Attempt to allocate a new stage.
* Pay special attention to possible race conditions.
*/
while ((new_stage = idetape_kmalloc_stage(tape)) == NULL) { while ((new_stage = idetape_kmalloc_stage(tape)) == NULL) {
spin_lock_irqsave(&tape->spinlock, flags); spin_lock_irqsave(&tape->lock, flags);
if (idetape_pipeline_active(tape)) { if (idetape_pipeline_active(tape)) {
idetape_wait_for_request(drive, tape->active_data_request); idetape_wait_for_request(drive, tape->active_data_rq);
spin_unlock_irqrestore(&tape->spinlock, flags); spin_unlock_irqrestore(&tape->lock, flags);
} else { } else {
spin_unlock_irqrestore(&tape->spinlock, flags); spin_unlock_irqrestore(&tape->lock, flags);
idetape_insert_pipeline_into_queue(drive); idetape_plug_pipeline(drive);
if (idetape_pipeline_active(tape)) if (idetape_pipeline_active(tape))
continue; continue;
/* /*
* Linux is short on memory. Fallback to * The machine is short on memory. Fallback to non-
* non-pipelined operation mode for this request. * pipelined operation mode for this request.
*/ */
return idetape_queue_rw_tail(drive, REQ_IDETAPE_WRITE, blocks, tape->merge_stage->bh); return idetape_queue_rw_tail(drive, REQ_IDETAPE_WRITE,
blocks, tape->merge_stage->bh);
} }
} }
rq = &new_stage->rq; rq = &new_stage->rq;
idetape_init_rq(rq, REQ_IDETAPE_WRITE); idetape_init_rq(rq, REQ_IDETAPE_WRITE);
/* Doesn't actually matter - We always assume sequential access */ /* Doesn't actually matter - We always assume sequential access */
rq->sector = tape->first_frame_position; rq->sector = tape->first_frame;
rq->nr_sectors = rq->current_nr_sectors = blocks; rq->current_nr_sectors = blocks;
rq->nr_sectors = blocks;
idetape_switch_buffers(tape, new_stage); idetape_switch_buffers(tape, new_stage);
idetape_add_stage_tail(drive, new_stage); idetape_add_stage_tail(drive, new_stage);
tape->pipeline_head++; tape->pipeline_head++;
calculate_speeds(drive); idetape_calculate_speeds(drive);
/* /*
* Estimate whether the tape has stopped writing by checking * Estimate whether the tape has stopped writing by checking if our
* if our write pipeline is currently empty. If we are not * write pipeline is currently empty. If we are not writing anymore,
* writing anymore, wait for the pipeline to be full enough * wait for the pipeline to be almost completely full (90%) before
* (90%) before starting to service requests, so that we will * starting to service requests, so that we will be able to keep up with
* be able to keep up with the higher speeds of the tape. * the higher speeds of the tape.
*/ */
if (!idetape_pipeline_active(tape)) { if (!idetape_pipeline_active(tape)) {
if (tape->nr_stages >= tape->max_stages * 9 / 10 || if (tape->nr_stages >= tape->max_stages * 9 / 10 ||
tape->nr_stages >= tape->max_stages - tape->uncontrolled_pipeline_head_speed * 3 * 1024 / tape->tape_block_size) { tape->nr_stages >= tape->max_stages -
tape->uncontrolled_pipeline_head_speed * 3 * 1024 /
tape->blk_size) {
tape->measure_insert_time = 1; tape->measure_insert_time = 1;
tape->insert_time = jiffies; tape->insert_time = jiffies;
tape->insert_size = 0; tape->insert_size = 0;
tape->insert_speed = 0; tape->insert_speed = 0;
idetape_insert_pipeline_into_queue(drive); idetape_plug_pipeline(drive);
} }
} }
if (test_and_clear_bit(IDETAPE_PIPELINE_ERROR, &tape->flags)) if (test_and_clear_bit(IDETAPE_PIPELINE_ERROR, &tape->flags))
...@@ -2534,31 +2367,32 @@ static int idetape_add_chrdev_write_request (ide_drive_t *drive, int blocks) ...@@ -2534,31 +2367,32 @@ static int idetape_add_chrdev_write_request (ide_drive_t *drive, int blocks)
} }
/* /*
* idetape_wait_for_pipeline will wait until all pending pipeline * Wait until all pending pipeline requests are serviced. Typically called on
* requests are serviced. Typically called on device close. * device close.
*/ */
static void idetape_wait_for_pipeline (ide_drive_t *drive) static void idetape_wait_for_pipeline(ide_drive_t *drive)
{ {
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
unsigned long flags; unsigned long flags;
while (tape->next_stage || idetape_pipeline_active(tape)) { while (tape->next_stage || idetape_pipeline_active(tape)) {
idetape_insert_pipeline_into_queue(drive); idetape_plug_pipeline(drive);
spin_lock_irqsave(&tape->spinlock, flags); spin_lock_irqsave(&tape->lock, flags);
if (idetape_pipeline_active(tape)) if (idetape_pipeline_active(tape))
idetape_wait_for_request(drive, tape->active_data_request); idetape_wait_for_request(drive, tape->active_data_rq);
spin_unlock_irqrestore(&tape->spinlock, flags); spin_unlock_irqrestore(&tape->lock, flags);
} }
} }
static void idetape_empty_write_pipeline (ide_drive_t *drive) static void idetape_empty_write_pipeline(ide_drive_t *drive)
{ {
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
int blocks, min; int blocks, min;
struct idetape_bh *bh; struct idetape_bh *bh;
if (tape->chrdev_direction != idetape_direction_write) { if (tape->chrdev_dir != IDETAPE_DIR_WRITE) {
printk(KERN_ERR "ide-tape: bug: Trying to empty write pipeline, but we are not writing.\n"); printk(KERN_ERR "ide-tape: bug: Trying to empty write pipeline,"
" but we are not writing.\n");
return; return;
} }
if (tape->merge_stage_size > tape->stage_size) { if (tape->merge_stage_size > tape->stage_size) {
...@@ -2566,12 +2400,13 @@ static void idetape_empty_write_pipeline (ide_drive_t *drive) ...@@ -2566,12 +2400,13 @@ static void idetape_empty_write_pipeline (ide_drive_t *drive)
tape->merge_stage_size = tape->stage_size; tape->merge_stage_size = tape->stage_size;
} }
if (tape->merge_stage_size) { if (tape->merge_stage_size) {
blocks = tape->merge_stage_size / tape->tape_block_size; blocks = tape->merge_stage_size / tape->blk_size;
if (tape->merge_stage_size % tape->tape_block_size) { if (tape->merge_stage_size % tape->blk_size) {
unsigned int i; unsigned int i;
blocks++; blocks++;
i = tape->tape_block_size - tape->merge_stage_size % tape->tape_block_size; i = tape->blk_size - tape->merge_stage_size %
tape->blk_size;
bh = tape->bh->b_reqnext; bh = tape->bh->b_reqnext;
while (bh) { while (bh) {
atomic_set(&bh->b_count, 0); atomic_set(&bh->b_count, 0);
...@@ -2580,12 +2415,14 @@ static void idetape_empty_write_pipeline (ide_drive_t *drive) ...@@ -2580,12 +2415,14 @@ static void idetape_empty_write_pipeline (ide_drive_t *drive)
bh = tape->bh; bh = tape->bh;
while (i) { while (i) {
if (bh == NULL) { if (bh == NULL) {
printk(KERN_INFO "ide-tape: bug,"
printk(KERN_INFO "ide-tape: bug, bh NULL\n"); " bh NULL\n");
break; break;
} }
min = min(i, (unsigned int)(bh->b_size - atomic_read(&bh->b_count))); min = min(i, (unsigned int)(bh->b_size -
memset(bh->b_data + atomic_read(&bh->b_count), 0, min); atomic_read(&bh->b_count)));
memset(bh->b_data + atomic_read(&bh->b_count),
0, min);
atomic_add(min, &bh->b_count); atomic_add(min, &bh->b_count);
i -= min; i -= min;
bh = bh->b_reqnext; bh = bh->b_reqnext;
...@@ -2600,13 +2437,13 @@ static void idetape_empty_write_pipeline (ide_drive_t *drive) ...@@ -2600,13 +2437,13 @@ static void idetape_empty_write_pipeline (ide_drive_t *drive)
tape->merge_stage = NULL; tape->merge_stage = NULL;
} }
clear_bit(IDETAPE_PIPELINE_ERROR, &tape->flags); clear_bit(IDETAPE_PIPELINE_ERROR, &tape->flags);
tape->chrdev_direction = idetape_direction_none; tape->chrdev_dir = IDETAPE_DIR_NONE;
/* /*
* On the next backup, perform the feedback loop again. * On the next backup, perform the feedback loop again. (I don't want to
* (I don't want to keep sense information between backups, * keep sense information between backups, as some systems are
* as some systems are constantly on, and the system load * constantly on, and the system load can be totally different on the
* can be totally different on the next backup). * next backup).
*/ */
tape->max_stages = tape->min_pipeline; tape->max_stages = tape->min_pipeline;
if (tape->first_stage != NULL || if (tape->first_stage != NULL ||
...@@ -2621,21 +2458,25 @@ static void idetape_empty_write_pipeline (ide_drive_t *drive) ...@@ -2621,21 +2458,25 @@ static void idetape_empty_write_pipeline (ide_drive_t *drive)
} }
} }
static void idetape_restart_speed_control (ide_drive_t *drive) static void idetape_restart_speed_control(ide_drive_t *drive)
{ {
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
tape->restart_speed_control_req = 0; tape->restart_speed_control_req = 0;
tape->pipeline_head = 0; tape->pipeline_head = 0;
tape->controlled_last_pipeline_head = tape->uncontrolled_last_pipeline_head = 0; tape->controlled_last_pipeline_head = 0;
tape->controlled_previous_pipeline_head = tape->uncontrolled_previous_pipeline_head = 0; tape->controlled_previous_pipeline_head = 0;
tape->pipeline_head_speed = tape->controlled_pipeline_head_speed = 5000; tape->uncontrolled_previous_pipeline_head = 0;
tape->controlled_pipeline_head_speed = 5000;
tape->pipeline_head_speed = 5000;
tape->uncontrolled_pipeline_head_speed = 0; tape->uncontrolled_pipeline_head_speed = 0;
tape->controlled_pipeline_head_time = tape->uncontrolled_pipeline_head_time = jiffies; tape->controlled_pipeline_head_time =
tape->controlled_previous_head_time = tape->uncontrolled_previous_head_time = jiffies; tape->uncontrolled_pipeline_head_time = jiffies;
tape->controlled_previous_head_time =
tape->uncontrolled_previous_head_time = jiffies;
} }
static int idetape_initiate_read (ide_drive_t *drive, int max_stages) static int idetape_init_read(ide_drive_t *drive, int max_stages)
{ {
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
idetape_stage_t *new_stage; idetape_stage_t *new_stage;
...@@ -2644,32 +2485,35 @@ static int idetape_initiate_read (ide_drive_t *drive, int max_stages) ...@@ -2644,32 +2485,35 @@ static int idetape_initiate_read (ide_drive_t *drive, int max_stages)
u16 blocks = *(u16 *)&tape->caps[12]; u16 blocks = *(u16 *)&tape->caps[12];
/* Initialize read operation */ /* Initialize read operation */
if (tape->chrdev_direction != idetape_direction_read) { if (tape->chrdev_dir != IDETAPE_DIR_READ) {
if (tape->chrdev_direction == idetape_direction_write) { if (tape->chrdev_dir == IDETAPE_DIR_WRITE) {
idetape_empty_write_pipeline(drive); idetape_empty_write_pipeline(drive);
idetape_flush_tape_buffers(drive); idetape_flush_tape_buffers(drive);
} }
if (tape->merge_stage || tape->merge_stage_size) { if (tape->merge_stage || tape->merge_stage_size) {
printk (KERN_ERR "ide-tape: merge_stage_size should be 0 now\n"); printk(KERN_ERR "ide-tape: merge_stage_size should be"
" 0 now\n");
tape->merge_stage_size = 0; tape->merge_stage_size = 0;
} }
if ((tape->merge_stage = __idetape_kmalloc_stage(tape, 0, 0)) == NULL) tape->merge_stage = __idetape_kmalloc_stage(tape, 0, 0);
if (!tape->merge_stage)
return -ENOMEM; return -ENOMEM;
tape->chrdev_direction = idetape_direction_read; tape->chrdev_dir = IDETAPE_DIR_READ;
/* /*
* Issue a read 0 command to ensure that DSC handshake * Issue a read 0 command to ensure that DSC handshake is
* is switched from completion mode to buffer available * switched from completion mode to buffer available mode.
* mode. * No point in issuing this if DSC overlap isn't supported, some
* No point in issuing this if DSC overlap isn't supported, * drives (Seagate STT3401A) will return an error.
* some drives (Seagate STT3401A) will return an error.
*/ */
if (drive->dsc_overlap) { if (drive->dsc_overlap) {
bytes_read = idetape_queue_rw_tail(drive, REQ_IDETAPE_READ, 0, tape->merge_stage->bh); bytes_read = idetape_queue_rw_tail(drive,
REQ_IDETAPE_READ, 0,
tape->merge_stage->bh);
if (bytes_read < 0) { if (bytes_read < 0) {
__idetape_kfree_stage(tape->merge_stage); __idetape_kfree_stage(tape->merge_stage);
tape->merge_stage = NULL; tape->merge_stage = NULL;
tape->chrdev_direction = idetape_direction_none; tape->chrdev_dir = IDETAPE_DIR_NONE;
return bytes_read; return bytes_read;
} }
} }
...@@ -2677,8 +2521,9 @@ static int idetape_initiate_read (ide_drive_t *drive, int max_stages) ...@@ -2677,8 +2521,9 @@ static int idetape_initiate_read (ide_drive_t *drive, int max_stages)
if (tape->restart_speed_control_req) if (tape->restart_speed_control_req)
idetape_restart_speed_control(drive); idetape_restart_speed_control(drive);
idetape_init_rq(&rq, REQ_IDETAPE_READ); idetape_init_rq(&rq, REQ_IDETAPE_READ);
rq.sector = tape->first_frame_position; rq.sector = tape->first_frame;
rq.nr_sectors = rq.current_nr_sectors = blocks; rq.nr_sectors = blocks;
rq.current_nr_sectors = blocks;
if (!test_bit(IDETAPE_PIPELINE_ERROR, &tape->flags) && if (!test_bit(IDETAPE_PIPELINE_ERROR, &tape->flags) &&
tape->nr_stages < max_stages) { tape->nr_stages < max_stages) {
new_stage = idetape_kmalloc_stage(tape); new_stage = idetape_kmalloc_stage(tape);
...@@ -2696,50 +2541,43 @@ static int idetape_initiate_read (ide_drive_t *drive, int max_stages) ...@@ -2696,50 +2541,43 @@ static int idetape_initiate_read (ide_drive_t *drive, int max_stages)
tape->insert_time = jiffies; tape->insert_time = jiffies;
tape->insert_size = 0; tape->insert_size = 0;
tape->insert_speed = 0; tape->insert_speed = 0;
idetape_insert_pipeline_into_queue(drive); idetape_plug_pipeline(drive);
} }
} }
return 0; return 0;
} }
/* /*
* idetape_add_chrdev_read_request is called from idetape_chrdev_read * Called from idetape_chrdev_read() to service a character device read request
* to service a character device read request and add read-ahead * and add read-ahead requests to our pipeline.
* requests to our pipeline.
*/ */
static int idetape_add_chrdev_read_request (ide_drive_t *drive,int blocks) static int idetape_add_chrdev_read_request(ide_drive_t *drive, int blocks)
{ {
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
unsigned long flags; unsigned long flags;
struct request *rq_ptr; struct request *rq_ptr;
int bytes_read; int bytes_read;
#if IDETAPE_DEBUG_LOG debug_log(DBG_PROCS, "Enter %s, %d blocks\n", __func__, blocks);
if (tape->debug_level >= 4)
printk(KERN_INFO "ide-tape: Reached idetape_add_chrdev_read_request, %d blocks\n", blocks);
#endif /* IDETAPE_DEBUG_LOG */
/* /* If we are at a filemark, return a read length of 0 */
* If we are at a filemark, return a read length of 0
*/
if (test_bit(IDETAPE_FILEMARK, &tape->flags)) if (test_bit(IDETAPE_FILEMARK, &tape->flags))
return 0; return 0;
/* /* Wait for the next block to reach the head of the pipeline. */
* Wait for the next block to be available at the head idetape_init_read(drive, tape->max_stages);
* of the pipeline
*/
idetape_initiate_read(drive, tape->max_stages);
if (tape->first_stage == NULL) { if (tape->first_stage == NULL) {
if (test_bit(IDETAPE_PIPELINE_ERROR, &tape->flags)) if (test_bit(IDETAPE_PIPELINE_ERROR, &tape->flags))
return 0; return 0;
return idetape_queue_rw_tail(drive, REQ_IDETAPE_READ, blocks, tape->merge_stage->bh); return idetape_queue_rw_tail(drive, REQ_IDETAPE_READ, blocks,
tape->merge_stage->bh);
} }
idetape_wait_first_stage(drive); idetape_wait_first_stage(drive);
rq_ptr = &tape->first_stage->rq; rq_ptr = &tape->first_stage->rq;
bytes_read = tape->tape_block_size * (rq_ptr->nr_sectors - rq_ptr->current_nr_sectors); bytes_read = tape->blk_size * (rq_ptr->nr_sectors -
rq_ptr->nr_sectors = rq_ptr->current_nr_sectors = 0; rq_ptr->current_nr_sectors);
rq_ptr->nr_sectors = 0;
rq_ptr->current_nr_sectors = 0;
if (rq_ptr->errors == IDETAPE_ERROR_EOD) if (rq_ptr->errors == IDETAPE_ERROR_EOD)
return 0; return 0;
...@@ -2747,43 +2585,46 @@ static int idetape_add_chrdev_read_request (ide_drive_t *drive,int blocks) ...@@ -2747,43 +2585,46 @@ static int idetape_add_chrdev_read_request (ide_drive_t *drive,int blocks)
idetape_switch_buffers(tape, tape->first_stage); idetape_switch_buffers(tape, tape->first_stage);
if (rq_ptr->errors == IDETAPE_ERROR_FILEMARK) if (rq_ptr->errors == IDETAPE_ERROR_FILEMARK)
set_bit(IDETAPE_FILEMARK, &tape->flags); set_bit(IDETAPE_FILEMARK, &tape->flags);
spin_lock_irqsave(&tape->spinlock, flags); spin_lock_irqsave(&tape->lock, flags);
idetape_remove_stage_head(drive); idetape_remove_stage_head(drive);
spin_unlock_irqrestore(&tape->spinlock, flags); spin_unlock_irqrestore(&tape->lock, flags);
tape->pipeline_head++; tape->pipeline_head++;
calculate_speeds(drive); idetape_calculate_speeds(drive);
} }
if (bytes_read > blocks * tape->tape_block_size) { if (bytes_read > blocks * tape->blk_size) {
printk(KERN_ERR "ide-tape: bug: trying to return more bytes than requested\n"); printk(KERN_ERR "ide-tape: bug: trying to return more bytes"
bytes_read = blocks * tape->tape_block_size; " than requested\n");
bytes_read = blocks * tape->blk_size;
} }
return (bytes_read); return (bytes_read);
} }
static void idetape_pad_zeros (ide_drive_t *drive, int bcount) static void idetape_pad_zeros(ide_drive_t *drive, int bcount)
{ {
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
struct idetape_bh *bh; struct idetape_bh *bh;
int blocks; int blocks;
while (bcount) { while (bcount) {
unsigned int count; unsigned int count;
bh = tape->merge_stage->bh; bh = tape->merge_stage->bh;
count = min(tape->stage_size, bcount); count = min(tape->stage_size, bcount);
bcount -= count; bcount -= count;
blocks = count / tape->tape_block_size; blocks = count / tape->blk_size;
while (count) { while (count) {
atomic_set(&bh->b_count, min(count, (unsigned int)bh->b_size)); atomic_set(&bh->b_count,
min(count, (unsigned int)bh->b_size));
memset(bh->b_data, 0, atomic_read(&bh->b_count)); memset(bh->b_data, 0, atomic_read(&bh->b_count));
count -= atomic_read(&bh->b_count); count -= atomic_read(&bh->b_count);
bh = bh->b_reqnext; bh = bh->b_reqnext;
} }
idetape_queue_rw_tail(drive, REQ_IDETAPE_WRITE, blocks, tape->merge_stage->bh); idetape_queue_rw_tail(drive, REQ_IDETAPE_WRITE, blocks,
tape->merge_stage->bh);
} }
} }
static int idetape_pipeline_size (ide_drive_t *drive) static int idetape_pipeline_size(ide_drive_t *drive)
{ {
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
idetape_stage_t *stage; idetape_stage_t *stage;
...@@ -2794,9 +2635,10 @@ static int idetape_pipeline_size (ide_drive_t *drive) ...@@ -2794,9 +2635,10 @@ static int idetape_pipeline_size (ide_drive_t *drive)
stage = tape->first_stage; stage = tape->first_stage;
while (stage != NULL) { while (stage != NULL) {
rq = &stage->rq; rq = &stage->rq;
size += tape->tape_block_size * (rq->nr_sectors-rq->current_nr_sectors); size += tape->blk_size * (rq->nr_sectors -
rq->current_nr_sectors);
if (rq->errors == IDETAPE_ERROR_FILEMARK) if (rq->errors == IDETAPE_ERROR_FILEMARK)
size += tape->tape_block_size; size += tape->blk_size;
stage = stage->next; stage = stage->next;
} }
size += tape->merge_stage_size; size += tape->merge_stage_size;
...@@ -2804,20 +2646,18 @@ static int idetape_pipeline_size (ide_drive_t *drive) ...@@ -2804,20 +2646,18 @@ static int idetape_pipeline_size (ide_drive_t *drive)
} }
/* /*
* Rewinds the tape to the Beginning Of the current Partition (BOP). * Rewinds the tape to the Beginning Of the current Partition (BOP). We
* * currently support only one partition.
* We currently support only one partition. */
*/ static int idetape_rewind_tape(ide_drive_t *drive)
static int idetape_rewind_tape (ide_drive_t *drive)
{ {
int retval; int retval;
idetape_pc_t pc; idetape_pc_t pc;
#if IDETAPE_DEBUG_LOG idetape_tape_t *tape;
idetape_tape_t *tape = drive->driver_data; tape = drive->driver_data;
if (tape->debug_level >= 2)
printk(KERN_INFO "ide-tape: Reached idetape_rewind_tape\n"); debug_log(DBG_SENSE, "Enter %s\n", __func__);
#endif /* IDETAPE_DEBUG_LOG */
idetape_create_rewind_cmd(drive, &pc); idetape_create_rewind_cmd(drive, &pc);
retval = idetape_queue_pc_tail(drive, &pc); retval = idetape_queue_pc_tail(drive, &pc);
if (retval) if (retval)
...@@ -2830,14 +2670,9 @@ static int idetape_rewind_tape (ide_drive_t *drive) ...@@ -2830,14 +2670,9 @@ static int idetape_rewind_tape (ide_drive_t *drive)
return 0; return 0;
} }
/* /* mtio.h compatible commands should be issued to the chrdev interface. */
* Our special ide-tape ioctl's. static int idetape_blkdev_ioctl(ide_drive_t *drive, unsigned int cmd,
* unsigned long arg)
* Currently there aren't any ioctl's.
* mtio.h compatible commands should be issued to the character device
* interface.
*/
static int idetape_blkdev_ioctl(ide_drive_t *drive, unsigned int cmd, unsigned long arg)
{ {
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
void __user *argp = (void __user *)arg; void __user *argp = (void __user *)arg;
...@@ -2848,44 +2683,41 @@ static int idetape_blkdev_ioctl(ide_drive_t *drive, unsigned int cmd, unsigned l ...@@ -2848,44 +2683,41 @@ static int idetape_blkdev_ioctl(ide_drive_t *drive, unsigned int cmd, unsigned l
int nr_stages; int nr_stages;
} config; } config;
#if IDETAPE_DEBUG_LOG debug_log(DBG_PROCS, "Enter %s\n", __func__);
if (tape->debug_level >= 4)
printk(KERN_INFO "ide-tape: Reached idetape_blkdev_ioctl\n");
#endif /* IDETAPE_DEBUG_LOG */
switch (cmd) { switch (cmd) {
case 0x0340: case 0x0340:
if (copy_from_user(&config, argp, sizeof(config))) if (copy_from_user(&config, argp, sizeof(config)))
return -EFAULT; return -EFAULT;
tape->best_dsc_rw_frequency = config.dsc_rw_frequency; tape->best_dsc_rw_freq = config.dsc_rw_frequency;
tape->max_stages = config.nr_stages; tape->max_stages = config.nr_stages;
break; break;
case 0x0350: case 0x0350:
config.dsc_rw_frequency = (int) tape->best_dsc_rw_frequency; config.dsc_rw_frequency = (int) tape->best_dsc_rw_freq;
config.nr_stages = tape->max_stages; config.nr_stages = tape->max_stages;
if (copy_to_user(argp, &config, sizeof(config))) if (copy_to_user(argp, &config, sizeof(config)))
return -EFAULT; return -EFAULT;
break; break;
default: default:
return -EIO; return -EIO;
} }
return 0; return 0;
} }
/* /*
* idetape_space_over_filemarks is now a bit more complicated than just * The function below is now a bit more complicated than just passing the
* passing the command to the tape since we may have crossed some * command to the tape since we may have crossed some filemarks during our
* filemarks during our pipelined read-ahead mode. * pipelined read-ahead mode. As a minor side effect, the pipeline enables us to
* * support MTFSFM when the filemark is in our internal pipeline even if the tape
* As a minor side effect, the pipeline enables us to support MTFSFM when * doesn't support spacing over filemarks in the reverse direction.
* the filemark is in our internal pipeline even if the tape doesn't
* support spacing over filemarks in the reverse direction.
*/ */
static int idetape_space_over_filemarks (ide_drive_t *drive,short mt_op,int mt_count) static int idetape_space_over_filemarks(ide_drive_t *drive, short mt_op,
int mt_count)
{ {
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
idetape_pc_t pc; idetape_pc_t pc;
unsigned long flags; unsigned long flags;
int retval,count=0; int retval, count = 0;
int sprev = !!(tape->caps[4] & 0x20); int sprev = !!(tape->caps[4] & 0x20);
if (mt_count == 0) if (mt_count == 0)
...@@ -2893,14 +2725,11 @@ static int idetape_space_over_filemarks (ide_drive_t *drive,short mt_op,int mt_c ...@@ -2893,14 +2725,11 @@ static int idetape_space_over_filemarks (ide_drive_t *drive,short mt_op,int mt_c
if (MTBSF == mt_op || MTBSFM == mt_op) { if (MTBSF == mt_op || MTBSFM == mt_op) {
if (!sprev) if (!sprev)
return -EIO; return -EIO;
mt_count = - mt_count; mt_count = -mt_count;
} }
if (tape->chrdev_direction == idetape_direction_read) { if (tape->chrdev_dir == IDETAPE_DIR_READ) {
/* /* its a read-ahead buffer, scan it for crossed filemarks. */
* We have a read-ahead buffer. Scan it for crossed
* filemarks.
*/
tape->merge_stage_size = 0; tape->merge_stage_size = 0;
if (test_and_clear_bit(IDETAPE_FILEMARK, &tape->flags)) if (test_and_clear_bit(IDETAPE_FILEMARK, &tape->flags))
++count; ++count;
...@@ -2910,24 +2739,27 @@ static int idetape_space_over_filemarks (ide_drive_t *drive,short mt_op,int mt_c ...@@ -2910,24 +2739,27 @@ static int idetape_space_over_filemarks (ide_drive_t *drive,short mt_op,int mt_c
set_bit(IDETAPE_FILEMARK, &tape->flags); set_bit(IDETAPE_FILEMARK, &tape->flags);
return 0; return 0;
} }
spin_lock_irqsave(&tape->spinlock, flags); spin_lock_irqsave(&tape->lock, flags);
if (tape->first_stage == tape->active_stage) { if (tape->first_stage == tape->active_stage) {
/* /*
* We have reached the active stage in the read pipeline. * We have reached the active stage in the read
* There is no point in allowing the drive to continue * pipeline. There is no point in allowing the
* reading any farther, so we stop the pipeline. * drive to continue reading any farther, so we
* stop the pipeline.
* *
* This section should be moved to a separate subroutine, * This section should be moved to a separate
* because a similar function is performed in * subroutine because similar operations are
* __idetape_discard_read_pipeline(), for example. * done in __idetape_discard_read_pipeline(),
* for example.
*/ */
tape->next_stage = NULL; tape->next_stage = NULL;
spin_unlock_irqrestore(&tape->spinlock, flags); spin_unlock_irqrestore(&tape->lock, flags);
idetape_wait_first_stage(drive); idetape_wait_first_stage(drive);
tape->next_stage = tape->first_stage->next; tape->next_stage = tape->first_stage->next;
} else } else
spin_unlock_irqrestore(&tape->spinlock, flags); spin_unlock_irqrestore(&tape->lock, flags);
if (tape->first_stage->rq.errors == IDETAPE_ERROR_FILEMARK) if (tape->first_stage->rq.errors ==
IDETAPE_ERROR_FILEMARK)
++count; ++count;
idetape_remove_stage_head(drive); idetape_remove_stage_head(drive);
} }
...@@ -2935,73 +2767,74 @@ static int idetape_space_over_filemarks (ide_drive_t *drive,short mt_op,int mt_c ...@@ -2935,73 +2767,74 @@ static int idetape_space_over_filemarks (ide_drive_t *drive,short mt_op,int mt_c
} }
/* /*
* The filemark was not found in our internal pipeline. * The filemark was not found in our internal pipeline; now we can issue
* Now we can issue the space command. * the space command.
*/ */
switch (mt_op) { switch (mt_op) {
case MTFSF: case MTFSF:
case MTBSF: case MTBSF:
idetape_create_space_cmd(&pc,mt_count-count,IDETAPE_SPACE_OVER_FILEMARK); idetape_create_space_cmd(&pc, mt_count - count,
return (idetape_queue_pc_tail(drive, &pc)); IDETAPE_SPACE_OVER_FILEMARK);
case MTFSFM: return idetape_queue_pc_tail(drive, &pc);
case MTBSFM: case MTFSFM:
if (!sprev) case MTBSFM:
return (-EIO); if (!sprev)
retval = idetape_space_over_filemarks(drive, MTFSF, mt_count-count); return -EIO;
if (retval) return (retval); retval = idetape_space_over_filemarks(drive, MTFSF,
count = (MTBSFM == mt_op ? 1 : -1); mt_count - count);
return (idetape_space_over_filemarks(drive, MTFSF, count)); if (retval)
default: return retval;
printk(KERN_ERR "ide-tape: MTIO operation %d not supported\n",mt_op); count = (MTBSFM == mt_op ? 1 : -1);
return (-EIO); return idetape_space_over_filemarks(drive, MTFSF, count);
default:
printk(KERN_ERR "ide-tape: MTIO operation %d not supported\n",
mt_op);
return -EIO;
} }
} }
/* /*
* Our character device read / write functions. * Our character device read / write functions.
* *
* The tape is optimized to maximize throughput when it is transferring * The tape is optimized to maximize throughput when it is transferring an
* an integral number of the "continuous transfer limit", which is * integral number of the "continuous transfer limit", which is a parameter of
* a parameter of the specific tape (26 KB on my particular tape). * the specific tape (26kB on my particular tape, 32kB for Onstream).
* (32 kB for Onstream)
* *
* As of version 1.3 of the driver, the character device provides an * As of version 1.3 of the driver, the character device provides an abstract
* abstract continuous view of the media - any mix of block sizes (even 1 * continuous view of the media - any mix of block sizes (even 1 byte) on the
* byte) on the same backup/restore procedure is supported. The driver * same backup/restore procedure is supported. The driver will internally
* will internally convert the requests to the recommended transfer unit, * convert the requests to the recommended transfer unit, so that an unmatch
* so that an unmatch between the user's block size to the recommended * between the user's block size to the recommended size will only result in a
* size will only result in a (slightly) increased driver overhead, but * (slightly) increased driver overhead, but will no longer hit performance.
* will no longer hit performance. * This is not applicable to Onstream.
* This is not applicable to Onstream.
*/ */
static ssize_t idetape_chrdev_read (struct file *file, char __user *buf, static ssize_t idetape_chrdev_read(struct file *file, char __user *buf,
size_t count, loff_t *ppos) size_t count, loff_t *ppos)
{ {
struct ide_tape_obj *tape = ide_tape_f(file); struct ide_tape_obj *tape = ide_tape_f(file);
ide_drive_t *drive = tape->drive; ide_drive_t *drive = tape->drive;
ssize_t bytes_read,temp, actually_read = 0, rc; ssize_t bytes_read, temp, actually_read = 0, rc;
ssize_t ret = 0; ssize_t ret = 0;
u16 ctl = *(u16 *)&tape->caps[12]; u16 ctl = *(u16 *)&tape->caps[12];
#if IDETAPE_DEBUG_LOG debug_log(DBG_CHRDEV, "Enter %s, count %Zd\n", __func__, count);
if (tape->debug_level >= 3)
printk(KERN_INFO "ide-tape: Reached idetape_chrdev_read, count %Zd\n", count);
#endif /* IDETAPE_DEBUG_LOG */
if (tape->chrdev_direction != idetape_direction_read) { if (tape->chrdev_dir != IDETAPE_DIR_READ) {
if (test_bit(IDETAPE_DETECT_BS, &tape->flags)) if (test_bit(IDETAPE_DETECT_BS, &tape->flags))
if (count > tape->tape_block_size && if (count > tape->blk_size &&
(count % tape->tape_block_size) == 0) (count % tape->blk_size) == 0)
tape->user_bs_factor = count / tape->tape_block_size; tape->user_bs_factor = count / tape->blk_size;
} }
if ((rc = idetape_initiate_read(drive, tape->max_stages)) < 0) rc = idetape_init_read(drive, tape->max_stages);
if (rc < 0)
return rc; return rc;
if (count == 0) if (count == 0)
return (0); return (0);
if (tape->merge_stage_size) { if (tape->merge_stage_size) {
actually_read = min((unsigned int)(tape->merge_stage_size), (unsigned int)count); actually_read = min((unsigned int)(tape->merge_stage_size),
if (idetape_copy_stage_to_user(tape, buf, tape->merge_stage, actually_read)) (unsigned int)count);
if (idetape_copy_stage_to_user(tape, buf, tape->merge_stage,
actually_read))
ret = -EFAULT; ret = -EFAULT;
buf += actually_read; buf += actually_read;
tape->merge_stage_size -= actually_read; tape->merge_stage_size -= actually_read;
...@@ -3011,7 +2844,8 @@ static ssize_t idetape_chrdev_read (struct file *file, char __user *buf, ...@@ -3011,7 +2844,8 @@ static ssize_t idetape_chrdev_read (struct file *file, char __user *buf,
bytes_read = idetape_add_chrdev_read_request(drive, ctl); bytes_read = idetape_add_chrdev_read_request(drive, ctl);
if (bytes_read <= 0) if (bytes_read <= 0)
goto finish; goto finish;
if (idetape_copy_stage_to_user(tape, buf, tape->merge_stage, bytes_read)) if (idetape_copy_stage_to_user(tape, buf, tape->merge_stage,
bytes_read))
ret = -EFAULT; ret = -EFAULT;
buf += bytes_read; buf += bytes_read;
count -= bytes_read; count -= bytes_read;
...@@ -3022,25 +2856,24 @@ static ssize_t idetape_chrdev_read (struct file *file, char __user *buf, ...@@ -3022,25 +2856,24 @@ static ssize_t idetape_chrdev_read (struct file *file, char __user *buf,
if (bytes_read <= 0) if (bytes_read <= 0)
goto finish; goto finish;
temp = min((unsigned long)count, (unsigned long)bytes_read); temp = min((unsigned long)count, (unsigned long)bytes_read);
if (idetape_copy_stage_to_user(tape, buf, tape->merge_stage, temp)) if (idetape_copy_stage_to_user(tape, buf, tape->merge_stage,
temp))
ret = -EFAULT; ret = -EFAULT;
actually_read += temp; actually_read += temp;
tape->merge_stage_size = bytes_read-temp; tape->merge_stage_size = bytes_read-temp;
} }
finish: finish:
if (!actually_read && test_bit(IDETAPE_FILEMARK, &tape->flags)) { if (!actually_read && test_bit(IDETAPE_FILEMARK, &tape->flags)) {
#if IDETAPE_DEBUG_LOG debug_log(DBG_SENSE, "%s: spacing over filemark\n", tape->name);
if (tape->debug_level >= 2)
printk(KERN_INFO "ide-tape: %s: spacing over filemark\n", tape->name);
#endif
idetape_space_over_filemarks(drive, MTFSF, 1); idetape_space_over_filemarks(drive, MTFSF, 1);
return 0; return 0;
} }
return (ret) ? ret : actually_read; return ret ? ret : actually_read;
} }
static ssize_t idetape_chrdev_write (struct file *file, const char __user *buf, static ssize_t idetape_chrdev_write(struct file *file, const char __user *buf,
size_t count, loff_t *ppos) size_t count, loff_t *ppos)
{ {
struct ide_tape_obj *tape = ide_tape_f(file); struct ide_tape_obj *tape = ide_tape_f(file);
...@@ -3053,39 +2886,37 @@ static ssize_t idetape_chrdev_write (struct file *file, const char __user *buf, ...@@ -3053,39 +2886,37 @@ static ssize_t idetape_chrdev_write (struct file *file, const char __user *buf,
if (tape->write_prot) if (tape->write_prot)
return -EACCES; return -EACCES;
#if IDETAPE_DEBUG_LOG debug_log(DBG_CHRDEV, "Enter %s, count %Zd\n", __func__, count);
if (tape->debug_level >= 3)
printk(KERN_INFO "ide-tape: Reached idetape_chrdev_write, "
"count %Zd\n", count);
#endif /* IDETAPE_DEBUG_LOG */
/* Initialize write operation */ /* Initialize write operation */
if (tape->chrdev_direction != idetape_direction_write) { if (tape->chrdev_dir != IDETAPE_DIR_WRITE) {
if (tape->chrdev_direction == idetape_direction_read) if (tape->chrdev_dir == IDETAPE_DIR_READ)
idetape_discard_read_pipeline(drive, 1); idetape_discard_read_pipeline(drive, 1);
if (tape->merge_stage || tape->merge_stage_size) { if (tape->merge_stage || tape->merge_stage_size) {
printk(KERN_ERR "ide-tape: merge_stage_size " printk(KERN_ERR "ide-tape: merge_stage_size "
"should be 0 now\n"); "should be 0 now\n");
tape->merge_stage_size = 0; tape->merge_stage_size = 0;
} }
if ((tape->merge_stage = __idetape_kmalloc_stage(tape, 0, 0)) == NULL) tape->merge_stage = __idetape_kmalloc_stage(tape, 0, 0);
if (!tape->merge_stage)
return -ENOMEM; return -ENOMEM;
tape->chrdev_direction = idetape_direction_write; tape->chrdev_dir = IDETAPE_DIR_WRITE;
idetape_init_merge_stage(tape); idetape_init_merge_stage(tape);
/* /*
* Issue a write 0 command to ensure that DSC handshake * Issue a write 0 command to ensure that DSC handshake is
* is switched from completion mode to buffer available * switched from completion mode to buffer available mode. No
* mode. * point in issuing this if DSC overlap isn't supported, some
* No point in issuing this if DSC overlap isn't supported, * drives (Seagate STT3401A) will return an error.
* some drives (Seagate STT3401A) will return an error.
*/ */
if (drive->dsc_overlap) { if (drive->dsc_overlap) {
ssize_t retval = idetape_queue_rw_tail(drive, REQ_IDETAPE_WRITE, 0, tape->merge_stage->bh); ssize_t retval = idetape_queue_rw_tail(drive,
REQ_IDETAPE_WRITE, 0,
tape->merge_stage->bh);
if (retval < 0) { if (retval < 0) {
__idetape_kfree_stage(tape->merge_stage); __idetape_kfree_stage(tape->merge_stage);
tape->merge_stage = NULL; tape->merge_stage = NULL;
tape->chrdev_direction = idetape_direction_none; tape->chrdev_dir = IDETAPE_DIR_NONE;
return retval; return retval;
} }
} }
...@@ -3096,11 +2927,14 @@ static ssize_t idetape_chrdev_write (struct file *file, const char __user *buf, ...@@ -3096,11 +2927,14 @@ static ssize_t idetape_chrdev_write (struct file *file, const char __user *buf,
idetape_restart_speed_control(drive); idetape_restart_speed_control(drive);
if (tape->merge_stage_size) { if (tape->merge_stage_size) {
if (tape->merge_stage_size >= tape->stage_size) { if (tape->merge_stage_size >= tape->stage_size) {
printk(KERN_ERR "ide-tape: bug: merge buffer too big\n"); printk(KERN_ERR "ide-tape: bug: merge buf too big\n");
tape->merge_stage_size = 0; tape->merge_stage_size = 0;
} }
actually_written = min((unsigned int)(tape->stage_size - tape->merge_stage_size), (unsigned int)count); actually_written = min((unsigned int)
if (idetape_copy_stage_from_user(tape, tape->merge_stage, buf, actually_written)) (tape->stage_size - tape->merge_stage_size),
(unsigned int)count);
if (idetape_copy_stage_from_user(tape, tape->merge_stage, buf,
actually_written))
ret = -EFAULT; ret = -EFAULT;
buf += actually_written; buf += actually_written;
tape->merge_stage_size += actually_written; tape->merge_stage_size += actually_written;
...@@ -3116,7 +2950,8 @@ static ssize_t idetape_chrdev_write (struct file *file, const char __user *buf, ...@@ -3116,7 +2950,8 @@ static ssize_t idetape_chrdev_write (struct file *file, const char __user *buf,
} }
while (count >= tape->stage_size) { while (count >= tape->stage_size) {
ssize_t retval; ssize_t retval;
if (idetape_copy_stage_from_user(tape, tape->merge_stage, buf, tape->stage_size)) if (idetape_copy_stage_from_user(tape, tape->merge_stage, buf,
tape->stage_size))
ret = -EFAULT; ret = -EFAULT;
buf += tape->stage_size; buf += tape->stage_size;
count -= tape->stage_size; count -= tape->stage_size;
...@@ -3127,14 +2962,15 @@ static ssize_t idetape_chrdev_write (struct file *file, const char __user *buf, ...@@ -3127,14 +2962,15 @@ static ssize_t idetape_chrdev_write (struct file *file, const char __user *buf,
} }
if (count) { if (count) {
actually_written += count; actually_written += count;
if (idetape_copy_stage_from_user(tape, tape->merge_stage, buf, count)) if (idetape_copy_stage_from_user(tape, tape->merge_stage, buf,
count))
ret = -EFAULT; ret = -EFAULT;
tape->merge_stage_size += count; tape->merge_stage_size += count;
} }
return (ret) ? ret : actually_written; return ret ? ret : actually_written;
} }
static int idetape_write_filemark (ide_drive_t *drive) static int idetape_write_filemark(ide_drive_t *drive)
{ {
idetape_pc_t pc; idetape_pc_t pc;
...@@ -3165,113 +3001,117 @@ static int idetape_mtioctop(ide_drive_t *drive, short mt_op, int mt_count) ...@@ -3165,113 +3001,117 @@ static int idetape_mtioctop(ide_drive_t *drive, short mt_op, int mt_count)
{ {
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
idetape_pc_t pc; idetape_pc_t pc;
int i,retval; int i, retval;
#if IDETAPE_DEBUG_LOG debug_log(DBG_ERR, "Handling MTIOCTOP ioctl: mt_op=%d, mt_count=%d\n",
if (tape->debug_level >= 1) mt_op, mt_count);
printk(KERN_INFO "ide-tape: Handling MTIOCTOP ioctl: "
"mt_op=%d, mt_count=%d\n", mt_op, mt_count); /* Commands which need our pipelined read-ahead stages. */
#endif /* IDETAPE_DEBUG_LOG */
/*
* Commands which need our pipelined read-ahead stages.
*/
switch (mt_op) { switch (mt_op) {
case MTFSF: case MTFSF:
case MTFSFM: case MTFSFM:
case MTBSF: case MTBSF:
case MTBSFM: case MTBSFM:
if (!mt_count) if (!mt_count)
return (0); return 0;
return (idetape_space_over_filemarks(drive,mt_op,mt_count)); return idetape_space_over_filemarks(drive, mt_op, mt_count);
default: default:
break; break;
} }
switch (mt_op) { switch (mt_op) {
case MTWEOF: case MTWEOF:
if (tape->write_prot) if (tape->write_prot)
return -EACCES; return -EACCES;
idetape_discard_read_pipeline(drive, 1); idetape_discard_read_pipeline(drive, 1);
for (i = 0; i < mt_count; i++) { for (i = 0; i < mt_count; i++) {
retval = idetape_write_filemark(drive); retval = idetape_write_filemark(drive);
if (retval) if (retval)
return retval; return retval;
} }
return (0); return 0;
case MTREW: case MTREW:
idetape_discard_read_pipeline(drive, 0); idetape_discard_read_pipeline(drive, 0);
if (idetape_rewind_tape(drive)) if (idetape_rewind_tape(drive))
return -EIO;
return 0;
case MTLOAD:
idetape_discard_read_pipeline(drive, 0);
idetape_create_load_unload_cmd(drive, &pc,
IDETAPE_LU_LOAD_MASK);
return idetape_queue_pc_tail(drive, &pc);
case MTUNLOAD:
case MTOFFL:
/*
* If door is locked, attempt to unlock before
* attempting to eject.
*/
if (tape->door_locked) {
if (idetape_create_prevent_cmd(drive, &pc, 0))
if (!idetape_queue_pc_tail(drive, &pc))
tape->door_locked = DOOR_UNLOCKED;
}
idetape_discard_read_pipeline(drive, 0);
idetape_create_load_unload_cmd(drive, &pc,
!IDETAPE_LU_LOAD_MASK);
retval = idetape_queue_pc_tail(drive, &pc);
if (!retval)
clear_bit(IDETAPE_MEDIUM_PRESENT, &tape->flags);
return retval;
case MTNOP:
idetape_discard_read_pipeline(drive, 0);
return idetape_flush_tape_buffers(drive);
case MTRETEN:
idetape_discard_read_pipeline(drive, 0);
idetape_create_load_unload_cmd(drive, &pc,
IDETAPE_LU_RETENSION_MASK | IDETAPE_LU_LOAD_MASK);
return idetape_queue_pc_tail(drive, &pc);
case MTEOM:
idetape_create_space_cmd(&pc, 0, IDETAPE_SPACE_TO_EOD);
return idetape_queue_pc_tail(drive, &pc);
case MTERASE:
(void)idetape_rewind_tape(drive);
idetape_create_erase_cmd(&pc);
return idetape_queue_pc_tail(drive, &pc);
case MTSETBLK:
if (mt_count) {
if (mt_count < tape->blk_size ||
mt_count % tape->blk_size)
return -EIO; return -EIO;
tape->user_bs_factor = mt_count / tape->blk_size;
clear_bit(IDETAPE_DETECT_BS, &tape->flags);
} else
set_bit(IDETAPE_DETECT_BS, &tape->flags);
return 0;
case MTSEEK:
idetape_discard_read_pipeline(drive, 0);
return idetape_position_tape(drive,
mt_count * tape->user_bs_factor, tape->partition, 0);
case MTSETPART:
idetape_discard_read_pipeline(drive, 0);
return idetape_position_tape(drive, 0, mt_count, 0);
case MTFSR:
case MTBSR:
case MTLOCK:
if (!idetape_create_prevent_cmd(drive, &pc, 1))
return 0; return 0;
case MTLOAD: retval = idetape_queue_pc_tail(drive, &pc);
idetape_discard_read_pipeline(drive, 0); if (retval)
idetape_create_load_unload_cmd(drive, &pc, IDETAPE_LU_LOAD_MASK);
return (idetape_queue_pc_tail(drive, &pc));
case MTUNLOAD:
case MTOFFL:
/*
* If door is locked, attempt to unlock before
* attempting to eject.
*/
if (tape->door_locked) {
if (idetape_create_prevent_cmd(drive, &pc, 0))
if (!idetape_queue_pc_tail(drive, &pc))
tape->door_locked = DOOR_UNLOCKED;
}
idetape_discard_read_pipeline(drive, 0);
idetape_create_load_unload_cmd(drive, &pc,!IDETAPE_LU_LOAD_MASK);
retval = idetape_queue_pc_tail(drive, &pc);
if (!retval)
clear_bit(IDETAPE_MEDIUM_PRESENT, &tape->flags);
return retval; return retval;
case MTNOP: tape->door_locked = DOOR_EXPLICITLY_LOCKED;
idetape_discard_read_pipeline(drive, 0); return 0;
return (idetape_flush_tape_buffers(drive)); case MTUNLOCK:
case MTRETEN: if (!idetape_create_prevent_cmd(drive, &pc, 0))
idetape_discard_read_pipeline(drive, 0);
idetape_create_load_unload_cmd(drive, &pc,IDETAPE_LU_RETENSION_MASK | IDETAPE_LU_LOAD_MASK);
return (idetape_queue_pc_tail(drive, &pc));
case MTEOM:
idetape_create_space_cmd(&pc, 0, IDETAPE_SPACE_TO_EOD);
return (idetape_queue_pc_tail(drive, &pc));
case MTERASE:
(void) idetape_rewind_tape(drive);
idetape_create_erase_cmd(&pc);
return (idetape_queue_pc_tail(drive, &pc));
case MTSETBLK:
if (mt_count) {
if (mt_count < tape->tape_block_size || mt_count % tape->tape_block_size)
return -EIO;
tape->user_bs_factor = mt_count / tape->tape_block_size;
clear_bit(IDETAPE_DETECT_BS, &tape->flags);
} else
set_bit(IDETAPE_DETECT_BS, &tape->flags);
return 0;
case MTSEEK:
idetape_discard_read_pipeline(drive, 0);
return idetape_position_tape(drive, mt_count * tape->user_bs_factor, tape->partition, 0);
case MTSETPART:
idetape_discard_read_pipeline(drive, 0);
return (idetape_position_tape(drive, 0, mt_count, 0));
case MTFSR:
case MTBSR:
case MTLOCK:
if (!idetape_create_prevent_cmd(drive, &pc, 1))
return 0;
retval = idetape_queue_pc_tail(drive, &pc);
if (retval) return retval;
tape->door_locked = DOOR_EXPLICITLY_LOCKED;
return 0;
case MTUNLOCK:
if (!idetape_create_prevent_cmd(drive, &pc, 0))
return 0;
retval = idetape_queue_pc_tail(drive, &pc);
if (retval) return retval;
tape->door_locked = DOOR_UNLOCKED;
return 0; return 0;
default: retval = idetape_queue_pc_tail(drive, &pc);
printk(KERN_ERR "ide-tape: MTIO operation %d not " if (retval)
"supported\n", mt_op); return retval;
return (-EIO); tape->door_locked = DOOR_UNLOCKED;
return 0;
default:
printk(KERN_ERR "ide-tape: MTIO operation %d not supported\n",
mt_op);
return -EIO;
} }
} }
...@@ -3288,50 +3128,51 @@ static int idetape_chrdev_ioctl(struct inode *inode, struct file *file, ...@@ -3288,50 +3128,51 @@ static int idetape_chrdev_ioctl(struct inode *inode, struct file *file,
struct mtop mtop; struct mtop mtop;
struct mtget mtget; struct mtget mtget;
struct mtpos mtpos; struct mtpos mtpos;
int block_offset = 0, position = tape->first_frame_position; int block_offset = 0, position = tape->first_frame;
void __user *argp = (void __user *)arg; void __user *argp = (void __user *)arg;
#if IDETAPE_DEBUG_LOG debug_log(DBG_CHRDEV, "Enter %s, cmd=%u\n", __func__, cmd);
if (tape->debug_level >= 3)
printk(KERN_INFO "ide-tape: Reached idetape_chrdev_ioctl, "
"cmd=%u\n", cmd);
#endif /* IDETAPE_DEBUG_LOG */
tape->restart_speed_control_req = 1; tape->restart_speed_control_req = 1;
if (tape->chrdev_direction == idetape_direction_write) { if (tape->chrdev_dir == IDETAPE_DIR_WRITE) {
idetape_empty_write_pipeline(drive); idetape_empty_write_pipeline(drive);
idetape_flush_tape_buffers(drive); idetape_flush_tape_buffers(drive);
} }
if (cmd == MTIOCGET || cmd == MTIOCPOS) { if (cmd == MTIOCGET || cmd == MTIOCPOS) {
block_offset = idetape_pipeline_size(drive) / (tape->tape_block_size * tape->user_bs_factor); block_offset = idetape_pipeline_size(drive) /
if ((position = idetape_read_position(drive)) < 0) (tape->blk_size * tape->user_bs_factor);
position = idetape_read_position(drive);
if (position < 0)
return -EIO; return -EIO;
} }
switch (cmd) { switch (cmd) {
case MTIOCTOP: case MTIOCTOP:
if (copy_from_user(&mtop, argp, sizeof (struct mtop))) if (copy_from_user(&mtop, argp, sizeof(struct mtop)))
return -EFAULT; return -EFAULT;
return (idetape_mtioctop(drive,mtop.mt_op,mtop.mt_count)); return idetape_mtioctop(drive, mtop.mt_op, mtop.mt_count);
case MTIOCGET: case MTIOCGET:
memset(&mtget, 0, sizeof (struct mtget)); memset(&mtget, 0, sizeof(struct mtget));
mtget.mt_type = MT_ISSCSI2; mtget.mt_type = MT_ISSCSI2;
mtget.mt_blkno = position / tape->user_bs_factor - block_offset; mtget.mt_blkno = position / tape->user_bs_factor - block_offset;
mtget.mt_dsreg = ((tape->tape_block_size * tape->user_bs_factor) << MT_ST_BLKSIZE_SHIFT) & MT_ST_BLKSIZE_MASK; mtget.mt_dsreg =
if (tape->drv_write_prot) { ((tape->blk_size * tape->user_bs_factor)
mtget.mt_gstat |= GMT_WR_PROT(0xffffffff); << MT_ST_BLKSIZE_SHIFT) & MT_ST_BLKSIZE_MASK;
}
if (copy_to_user(argp, &mtget, sizeof(struct mtget))) if (tape->drv_write_prot)
return -EFAULT; mtget.mt_gstat |= GMT_WR_PROT(0xffffffff);
return 0;
case MTIOCPOS: if (copy_to_user(argp, &mtget, sizeof(struct mtget)))
mtpos.mt_blkno = position / tape->user_bs_factor - block_offset; return -EFAULT;
if (copy_to_user(argp, &mtpos, sizeof(struct mtpos))) return 0;
return -EFAULT; case MTIOCPOS:
return 0; mtpos.mt_blkno = position / tape->user_bs_factor - block_offset;
default: if (copy_to_user(argp, &mtpos, sizeof(struct mtpos)))
if (tape->chrdev_direction == idetape_direction_read) return -EFAULT;
idetape_discard_read_pipeline(drive, 1); return 0;
return idetape_blkdev_ioctl(drive, cmd, arg); default:
if (tape->chrdev_dir == IDETAPE_DIR_READ)
idetape_discard_read_pipeline(drive, 1);
return idetape_blkdev_ioctl(drive, cmd, arg);
} }
} }
...@@ -3347,23 +3188,20 @@ static void ide_tape_get_bsize_from_bdesc(ide_drive_t *drive) ...@@ -3347,23 +3188,20 @@ static void ide_tape_get_bsize_from_bdesc(ide_drive_t *drive)
idetape_create_mode_sense_cmd(&pc, IDETAPE_BLOCK_DESCRIPTOR); idetape_create_mode_sense_cmd(&pc, IDETAPE_BLOCK_DESCRIPTOR);
if (idetape_queue_pc_tail(drive, &pc)) { if (idetape_queue_pc_tail(drive, &pc)) {
printk(KERN_ERR "ide-tape: Can't get block descriptor\n"); printk(KERN_ERR "ide-tape: Can't get block descriptor\n");
if (tape->tape_block_size == 0) { if (tape->blk_size == 0) {
printk(KERN_WARNING "ide-tape: Cannot deal with zero " printk(KERN_WARNING "ide-tape: Cannot deal with zero "
"block size, assuming 32k\n"); "block size, assuming 32k\n");
tape->tape_block_size = 32768; tape->blk_size = 32768;
} }
return; return;
} }
tape->tape_block_size = (pc.buffer[4 + 5] << 16) + tape->blk_size = (pc.buffer[4 + 5] << 16) +
(pc.buffer[4 + 6] << 8) + (pc.buffer[4 + 6] << 8) +
pc.buffer[4 + 7]; pc.buffer[4 + 7];
tape->drv_write_prot = (pc.buffer[2] & 0x80) >> 7; tape->drv_write_prot = (pc.buffer[2] & 0x80) >> 7;
} }
/* static int idetape_chrdev_open(struct inode *inode, struct file *filp)
* Our character device open function.
*/
static int idetape_chrdev_open (struct inode *inode, struct file *filp)
{ {
unsigned int minor = iminor(inode), i = minor & ~0xc0; unsigned int minor = iminor(inode), i = minor & ~0xc0;
ide_drive_t *drive; ide_drive_t *drive;
...@@ -3371,6 +3209,15 @@ static int idetape_chrdev_open (struct inode *inode, struct file *filp) ...@@ -3371,6 +3209,15 @@ static int idetape_chrdev_open (struct inode *inode, struct file *filp)
idetape_pc_t pc; idetape_pc_t pc;
int retval; int retval;
if (i >= MAX_HWIFS * MAX_DRIVES)
return -ENXIO;
tape = ide_tape_chrdev_get(i);
if (!tape)
return -ENXIO;
debug_log(DBG_CHRDEV, "Enter %s\n", __func__);
/* /*
* We really want to do nonseekable_open(inode, filp); here, but some * We really want to do nonseekable_open(inode, filp); here, but some
* versions of tar incorrectly call lseek on tapes and bail out if that * versions of tar incorrectly call lseek on tapes and bail out if that
...@@ -3378,16 +3225,6 @@ static int idetape_chrdev_open (struct inode *inode, struct file *filp) ...@@ -3378,16 +3225,6 @@ static int idetape_chrdev_open (struct inode *inode, struct file *filp)
*/ */
filp->f_mode &= ~(FMODE_PREAD | FMODE_PWRITE); filp->f_mode &= ~(FMODE_PREAD | FMODE_PWRITE);
#if IDETAPE_DEBUG_LOG
printk(KERN_INFO "ide-tape: Reached idetape_chrdev_open\n");
#endif /* IDETAPE_DEBUG_LOG */
if (i >= MAX_HWIFS * MAX_DRIVES)
return -ENXIO;
if (!(tape = ide_tape_chrdev_get(i)))
return -ENXIO;
drive = tape->drive; drive = tape->drive;
filp->private_data = tape; filp->private_data = tape;
...@@ -3408,7 +3245,7 @@ static int idetape_chrdev_open (struct inode *inode, struct file *filp) ...@@ -3408,7 +3245,7 @@ static int idetape_chrdev_open (struct inode *inode, struct file *filp)
if (!test_bit(IDETAPE_ADDRESS_VALID, &tape->flags)) if (!test_bit(IDETAPE_ADDRESS_VALID, &tape->flags))
(void)idetape_rewind_tape(drive); (void)idetape_rewind_tape(drive);
if (tape->chrdev_direction != idetape_direction_read) if (tape->chrdev_dir != IDETAPE_DIR_READ)
clear_bit(IDETAPE_PIPELINE_ERROR, &tape->flags); clear_bit(IDETAPE_PIPELINE_ERROR, &tape->flags);
/* Read block size and write protect status from drive. */ /* Read block size and write protect status from drive. */
...@@ -3430,10 +3267,8 @@ static int idetape_chrdev_open (struct inode *inode, struct file *filp) ...@@ -3430,10 +3267,8 @@ static int idetape_chrdev_open (struct inode *inode, struct file *filp)
} }
} }
/* /* Lock the tape drive door so user can't eject. */
* Lock the tape drive door so user can't eject. if (tape->chrdev_dir == IDETAPE_DIR_NONE) {
*/
if (tape->chrdev_direction == idetape_direction_none) {
if (idetape_create_prevent_cmd(drive, &pc, 1)) { if (idetape_create_prevent_cmd(drive, &pc, 1)) {
if (!idetape_queue_pc_tail(drive, &pc)) { if (!idetape_queue_pc_tail(drive, &pc)) {
if (tape->door_locked != DOOR_EXPLICITLY_LOCKED) if (tape->door_locked != DOOR_EXPLICITLY_LOCKED)
...@@ -3450,14 +3285,15 @@ static int idetape_chrdev_open (struct inode *inode, struct file *filp) ...@@ -3450,14 +3285,15 @@ static int idetape_chrdev_open (struct inode *inode, struct file *filp)
return retval; return retval;
} }
static void idetape_write_release (ide_drive_t *drive, unsigned int minor) static void idetape_write_release(ide_drive_t *drive, unsigned int minor)
{ {
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
idetape_empty_write_pipeline(drive); idetape_empty_write_pipeline(drive);
tape->merge_stage = __idetape_kmalloc_stage(tape, 1, 0); tape->merge_stage = __idetape_kmalloc_stage(tape, 1, 0);
if (tape->merge_stage != NULL) { if (tape->merge_stage != NULL) {
idetape_pad_zeros(drive, tape->tape_block_size * (tape->user_bs_factor - 1)); idetape_pad_zeros(drive, tape->blk_size *
(tape->user_bs_factor - 1));
__idetape_kfree_stage(tape->merge_stage); __idetape_kfree_stage(tape->merge_stage);
tape->merge_stage = NULL; tape->merge_stage = NULL;
} }
...@@ -3466,10 +3302,7 @@ static void idetape_write_release (ide_drive_t *drive, unsigned int minor) ...@@ -3466,10 +3302,7 @@ static void idetape_write_release (ide_drive_t *drive, unsigned int minor)
idetape_flush_tape_buffers(drive); idetape_flush_tape_buffers(drive);
} }
/* static int idetape_chrdev_release(struct inode *inode, struct file *filp)
* Our character device release function.
*/
static int idetape_chrdev_release (struct inode *inode, struct file *filp)
{ {
struct ide_tape_obj *tape = ide_tape_f(filp); struct ide_tape_obj *tape = ide_tape_f(filp);
ide_drive_t *drive = tape->drive; ide_drive_t *drive = tape->drive;
...@@ -3478,14 +3311,12 @@ static int idetape_chrdev_release (struct inode *inode, struct file *filp) ...@@ -3478,14 +3311,12 @@ static int idetape_chrdev_release (struct inode *inode, struct file *filp)
lock_kernel(); lock_kernel();
tape = drive->driver_data; tape = drive->driver_data;
#if IDETAPE_DEBUG_LOG
if (tape->debug_level >= 3)
printk(KERN_INFO "ide-tape: Reached idetape_chrdev_release\n");
#endif /* IDETAPE_DEBUG_LOG */
if (tape->chrdev_direction == idetape_direction_write) debug_log(DBG_CHRDEV, "Enter %s\n", __func__);
if (tape->chrdev_dir == IDETAPE_DIR_WRITE)
idetape_write_release(drive, minor); idetape_write_release(drive, minor);
if (tape->chrdev_direction == idetape_direction_read) { if (tape->chrdev_dir == IDETAPE_DIR_READ) {
if (minor < 128) if (minor < 128)
idetape_discard_read_pipeline(drive, 1); idetape_discard_read_pipeline(drive, 1);
else else
...@@ -3497,7 +3328,7 @@ static int idetape_chrdev_release (struct inode *inode, struct file *filp) ...@@ -3497,7 +3328,7 @@ static int idetape_chrdev_release (struct inode *inode, struct file *filp)
} }
if (minor < 128 && test_bit(IDETAPE_MEDIUM_PRESENT, &tape->flags)) if (minor < 128 && test_bit(IDETAPE_MEDIUM_PRESENT, &tape->flags))
(void) idetape_rewind_tape(drive); (void) idetape_rewind_tape(drive);
if (tape->chrdev_direction == idetape_direction_none) { if (tape->chrdev_dir == IDETAPE_DIR_NONE) {
if (tape->door_locked == DOOR_LOCKED) { if (tape->door_locked == DOOR_LOCKED) {
if (idetape_create_prevent_cmd(drive, &pc, 0)) { if (idetape_create_prevent_cmd(drive, &pc, 0)) {
if (!idetape_queue_pc_tail(drive, &pc)) if (!idetape_queue_pc_tail(drive, &pc))
...@@ -3512,37 +3343,39 @@ static int idetape_chrdev_release (struct inode *inode, struct file *filp) ...@@ -3512,37 +3343,39 @@ static int idetape_chrdev_release (struct inode *inode, struct file *filp)
} }
/* /*
* idetape_identify_device is called to check the contents of the * check the contents of the ATAPI IDENTIFY command results. We return:
* ATAPI IDENTIFY command results. We return:
* *
* 1 If the tape can be supported by us, based on the information * 1 - If the tape can be supported by us, based on the information we have so
* we have so far. * far.
* *
* 0 If this tape driver is not currently supported by us. * 0 - If this tape driver is not currently supported by us.
*/ */
static int idetape_identify_device (ide_drive_t *drive) static int idetape_identify_device(ide_drive_t *drive)
{ {
struct idetape_id_gcw gcw; u8 gcw[2], protocol, device_type, removable, packet_size;
struct hd_driveid *id = drive->id;
if (drive->id_read == 0) if (drive->id_read == 0)
return 1; return 1;
*((unsigned short *) &gcw) = id->config; *((unsigned short *) &gcw) = drive->id->config;
protocol = (gcw[1] & 0xC0) >> 6;
device_type = gcw[1] & 0x1F;
removable = !!(gcw[0] & 0x80);
packet_size = gcw[0] & 0x3;
/* Check that we can support this device */ /* Check that we can support this device */
if (protocol != 2)
if (gcw.protocol != 2)
printk(KERN_ERR "ide-tape: Protocol (0x%02x) is not ATAPI\n", printk(KERN_ERR "ide-tape: Protocol (0x%02x) is not ATAPI\n",
gcw.protocol); protocol);
else if (gcw.device_type != 1) else if (device_type != 1)
printk(KERN_ERR "ide-tape: Device type (0x%02x) is not set " printk(KERN_ERR "ide-tape: Device type (0x%02x) is not set "
"to tape\n", gcw.device_type); "to tape\n", device_type);
else if (!gcw.removable) else if (!removable)
printk(KERN_ERR "ide-tape: The removable flag is not set\n"); printk(KERN_ERR "ide-tape: The removable flag is not set\n");
else if (gcw.packet_size != 0) { else if (packet_size != 0) {
printk(KERN_ERR "ide-tape: Packet size (0x%02x) is not 12 " printk(KERN_ERR "ide-tape: Packet size (0x%02x) is not 12"
"bytes long\n", gcw.packet_size); " bytes\n", packet_size);
} else } else
return 1; return 1;
return 0; return 0;
...@@ -3550,9 +3383,9 @@ static int idetape_identify_device (ide_drive_t *drive) ...@@ -3550,9 +3383,9 @@ static int idetape_identify_device (ide_drive_t *drive)
static void idetape_get_inquiry_results(ide_drive_t *drive) static void idetape_get_inquiry_results(ide_drive_t *drive)
{ {
char *r;
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
idetape_pc_t pc; idetape_pc_t pc;
char fw_rev[6], vendor_id[10], product_id[18];
idetape_create_inquiry_cmd(&pc); idetape_create_inquiry_cmd(&pc);
if (idetape_queue_pc_tail(drive, &pc)) { if (idetape_queue_pc_tail(drive, &pc)) {
...@@ -3560,27 +3393,23 @@ static void idetape_get_inquiry_results(ide_drive_t *drive) ...@@ -3560,27 +3393,23 @@ static void idetape_get_inquiry_results(ide_drive_t *drive)
tape->name); tape->name);
return; return;
} }
memcpy(tape->vendor_id, &pc.buffer[8], 8); memcpy(vendor_id, &pc.buffer[8], 8);
memcpy(tape->product_id, &pc.buffer[16], 16); memcpy(product_id, &pc.buffer[16], 16);
memcpy(tape->firmware_revision, &pc.buffer[32], 4); memcpy(fw_rev, &pc.buffer[32], 4);
ide_fixstring(tape->vendor_id, 10, 0); ide_fixstring(vendor_id, 10, 0);
ide_fixstring(tape->product_id, 18, 0); ide_fixstring(product_id, 18, 0);
ide_fixstring(tape->firmware_revision, 6, 0); ide_fixstring(fw_rev, 6, 0);
r = tape->firmware_revision;
if (*(r + 1) == '.')
tape->firmware_revision_num = (*r - '0') * 100 +
(*(r + 2) - '0') * 10 + *(r + 3) - '0';
printk(KERN_INFO "ide-tape: %s <-> %s: %s %s rev %s\n", printk(KERN_INFO "ide-tape: %s <-> %s: %s %s rev %s\n",
drive->name, tape->name, tape->vendor_id, drive->name, tape->name, vendor_id, product_id, fw_rev);
tape->product_id, tape->firmware_revision);
} }
/* /*
* Ask the tape about its various parameters. In particular, we will adjust our * Ask the tape about its various parameters. In particular, we will adjust our
* data transfer buffer size to the recommended value as returned by the tape. * data transfer buffer size to the recommended value as returned by the tape.
*/ */
static void idetape_get_mode_sense_results (ide_drive_t *drive) static void idetape_get_mode_sense_results(ide_drive_t *drive)
{ {
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
idetape_pc_t pc; idetape_pc_t pc;
...@@ -3591,7 +3420,7 @@ static void idetape_get_mode_sense_results (ide_drive_t *drive) ...@@ -3591,7 +3420,7 @@ static void idetape_get_mode_sense_results (ide_drive_t *drive)
if (idetape_queue_pc_tail(drive, &pc)) { if (idetape_queue_pc_tail(drive, &pc)) {
printk(KERN_ERR "ide-tape: Can't get tape parameters - assuming" printk(KERN_ERR "ide-tape: Can't get tape parameters - assuming"
" some default values\n"); " some default values\n");
tape->tape_block_size = 512; tape->blk_size = 512;
put_unaligned(52, (u16 *)&tape->caps[12]); put_unaligned(52, (u16 *)&tape->caps[12]);
put_unaligned(540, (u16 *)&tape->caps[14]); put_unaligned(540, (u16 *)&tape->caps[14]);
put_unaligned(6*52, (u16 *)&tape->caps[16]); put_unaligned(6*52, (u16 *)&tape->caps[16]);
...@@ -3621,62 +3450,75 @@ static void idetape_get_mode_sense_results (ide_drive_t *drive) ...@@ -3621,62 +3450,75 @@ static void idetape_get_mode_sense_results (ide_drive_t *drive)
memcpy(&tape->caps, caps, 20); memcpy(&tape->caps, caps, 20);
if (caps[7] & 0x02) if (caps[7] & 0x02)
tape->tape_block_size = 512; tape->blk_size = 512;
else if (caps[7] & 0x04) else if (caps[7] & 0x04)
tape->tape_block_size = 1024; tape->blk_size = 1024;
} }
#ifdef CONFIG_IDE_PROC_FS #ifdef CONFIG_IDE_PROC_FS
static void idetape_add_settings (ide_drive_t *drive) static void idetape_add_settings(ide_drive_t *drive)
{ {
idetape_tape_t *tape = drive->driver_data; idetape_tape_t *tape = drive->driver_data;
/*
* drive setting name read/write data type min max mul_factor div_factor data pointer set function
*/
ide_add_setting(drive, "buffer", SETTING_READ, TYPE_SHORT, 0, 0xffff, ide_add_setting(drive, "buffer", SETTING_READ, TYPE_SHORT, 0, 0xffff,
1, 2, (u16 *)&tape->caps[16], NULL); 1, 2, (u16 *)&tape->caps[16], NULL);
ide_add_setting(drive, "pipeline_min", SETTING_RW, TYPE_INT, 1, 0xffff, tape->stage_size / 1024, 1, &tape->min_pipeline, NULL); ide_add_setting(drive, "pipeline_min", SETTING_RW, TYPE_INT, 1, 0xffff,
ide_add_setting(drive, "pipeline", SETTING_RW, TYPE_INT, 1, 0xffff, tape->stage_size / 1024, 1, &tape->max_stages, NULL); tape->stage_size / 1024, 1, &tape->min_pipeline, NULL);
ide_add_setting(drive, "pipeline_max", SETTING_RW, TYPE_INT, 1, 0xffff, tape->stage_size / 1024, 1, &tape->max_pipeline, NULL); ide_add_setting(drive, "pipeline", SETTING_RW, TYPE_INT, 1, 0xffff,
ide_add_setting(drive, "pipeline_used", SETTING_READ, TYPE_INT, 0, 0xffff, tape->stage_size / 1024, 1, &tape->nr_stages, NULL); tape->stage_size / 1024, 1, &tape->max_stages, NULL);
ide_add_setting(drive, "pipeline_pending", SETTING_READ, TYPE_INT, 0, 0xffff, tape->stage_size / 1024, 1, &tape->nr_pending_stages, NULL); ide_add_setting(drive, "pipeline_max", SETTING_RW, TYPE_INT, 1, 0xffff,
tape->stage_size / 1024, 1, &tape->max_pipeline, NULL);
ide_add_setting(drive, "pipeline_used", SETTING_READ, TYPE_INT, 0,
0xffff, tape->stage_size / 1024, 1, &tape->nr_stages,
NULL);
ide_add_setting(drive, "pipeline_pending", SETTING_READ, TYPE_INT, 0,
0xffff, tape->stage_size / 1024, 1,
&tape->nr_pending_stages, NULL);
ide_add_setting(drive, "speed", SETTING_READ, TYPE_SHORT, 0, 0xffff, ide_add_setting(drive, "speed", SETTING_READ, TYPE_SHORT, 0, 0xffff,
1, 1, (u16 *)&tape->caps[14], NULL); 1, 1, (u16 *)&tape->caps[14], NULL);
ide_add_setting(drive, "stage", SETTING_READ, TYPE_INT, 0, 0xffff, 1, 1024, &tape->stage_size, NULL); ide_add_setting(drive, "stage", SETTING_READ, TYPE_INT, 0, 0xffff, 1,
ide_add_setting(drive, "tdsc", SETTING_RW, TYPE_INT, IDETAPE_DSC_RW_MIN, IDETAPE_DSC_RW_MAX, 1000, HZ, &tape->best_dsc_rw_frequency, NULL); 1024, &tape->stage_size, NULL);
ide_add_setting(drive, "dsc_overlap", SETTING_RW, TYPE_BYTE, 0, 1, 1, 1, &drive->dsc_overlap, NULL); ide_add_setting(drive, "tdsc", SETTING_RW, TYPE_INT, IDETAPE_DSC_RW_MIN,
ide_add_setting(drive, "pipeline_head_speed_c",SETTING_READ, TYPE_INT, 0, 0xffff, 1, 1, &tape->controlled_pipeline_head_speed, NULL); IDETAPE_DSC_RW_MAX, 1000, HZ, &tape->best_dsc_rw_freq,
ide_add_setting(drive, "pipeline_head_speed_u",SETTING_READ, TYPE_INT, 0, 0xffff, 1, 1, &tape->uncontrolled_pipeline_head_speed,NULL); NULL);
ide_add_setting(drive, "avg_speed", SETTING_READ, TYPE_INT, 0, 0xffff, 1, 1, &tape->avg_speed, NULL); ide_add_setting(drive, "dsc_overlap", SETTING_RW, TYPE_BYTE, 0, 1, 1,
ide_add_setting(drive, "debug_level", SETTING_RW, TYPE_INT, 0, 0xffff, 1, 1, &tape->debug_level, NULL); 1, &drive->dsc_overlap, NULL);
ide_add_setting(drive, "pipeline_head_speed_c", SETTING_READ, TYPE_INT,
0, 0xffff, 1, 1, &tape->controlled_pipeline_head_speed,
NULL);
ide_add_setting(drive, "pipeline_head_speed_u", SETTING_READ, TYPE_INT,
0, 0xffff, 1, 1,
&tape->uncontrolled_pipeline_head_speed, NULL);
ide_add_setting(drive, "avg_speed", SETTING_READ, TYPE_INT, 0, 0xffff,
1, 1, &tape->avg_speed, NULL);
ide_add_setting(drive, "debug_mask", SETTING_RW, TYPE_INT, 0, 0xffff, 1,
1, &tape->debug_mask, NULL);
} }
#else #else
static inline void idetape_add_settings(ide_drive_t *drive) { ; } static inline void idetape_add_settings(ide_drive_t *drive) { ; }
#endif #endif
/* /*
* ide_setup is called to: * The function below is called to:
* *
* 1. Initialize our various state variables. * 1. Initialize our various state variables.
* 2. Ask the tape for its capabilities. * 2. Ask the tape for its capabilities.
* 3. Allocate a buffer which will be used for data * 3. Allocate a buffer which will be used for data transfer. The buffer size
* transfer. The buffer size is chosen based on * is chosen based on the recommendation which we received in step 2.
* the recommendation which we received in step (2).
* *
* Note that at this point ide.c already assigned us an irq, so that * Note that at this point ide.c already assigned us an irq, so that we can
* we can queue requests here and wait for their completion. * queue requests here and wait for their completion.
*/ */
static void idetape_setup (ide_drive_t *drive, idetape_tape_t *tape, int minor) static void idetape_setup(ide_drive_t *drive, idetape_tape_t *tape, int minor)
{ {
unsigned long t1, tmid, tn, t; unsigned long t1, tmid, tn, t;
int speed; int speed;
struct idetape_id_gcw gcw;
int stage_size; int stage_size;
u8 gcw[2];
struct sysinfo si; struct sysinfo si;
u16 *ctl = (u16 *)&tape->caps[12]; u16 *ctl = (u16 *)&tape->caps[12];
spin_lock_init(&tape->spinlock); spin_lock_init(&tape->lock);
drive->dsc_overlap = 1; drive->dsc_overlap = 1;
if (drive->hwif->host_flags & IDE_HFLAG_NO_DSC) { if (drive->hwif->host_flags & IDE_HFLAG_NO_DSC) {
printk(KERN_INFO "ide-tape: %s: disabling DSC overlap\n", printk(KERN_INFO "ide-tape: %s: disabling DSC overlap\n",
...@@ -3690,25 +3532,29 @@ static void idetape_setup (ide_drive_t *drive, idetape_tape_t *tape, int minor) ...@@ -3690,25 +3532,29 @@ static void idetape_setup (ide_drive_t *drive, idetape_tape_t *tape, int minor)
tape->name[0] = 'h'; tape->name[0] = 'h';
tape->name[1] = 't'; tape->name[1] = 't';
tape->name[2] = '0' + minor; tape->name[2] = '0' + minor;
tape->chrdev_direction = idetape_direction_none; tape->chrdev_dir = IDETAPE_DIR_NONE;
tape->pc = tape->pc_stack; tape->pc = tape->pc_stack;
tape->max_insert_speed = 10000; tape->max_insert_speed = 10000;
tape->speed_control = 1; tape->speed_control = 1;
*((unsigned short *) &gcw) = drive->id->config; *((unsigned short *) &gcw) = drive->id->config;
if (gcw.drq_type == 1)
/* Command packet DRQ type */
if (((gcw[0] & 0x60) >> 5) == 1)
set_bit(IDETAPE_DRQ_INTERRUPT, &tape->flags); set_bit(IDETAPE_DRQ_INTERRUPT, &tape->flags);
tape->min_pipeline = tape->max_pipeline = tape->max_stages = 10; tape->min_pipeline = 10;
tape->max_pipeline = 10;
tape->max_stages = 10;
idetape_get_inquiry_results(drive); idetape_get_inquiry_results(drive);
idetape_get_mode_sense_results(drive); idetape_get_mode_sense_results(drive);
ide_tape_get_bsize_from_bdesc(drive); ide_tape_get_bsize_from_bdesc(drive);
tape->user_bs_factor = 1; tape->user_bs_factor = 1;
tape->stage_size = *ctl * tape->tape_block_size; tape->stage_size = *ctl * tape->blk_size;
while (tape->stage_size > 0xffff) { while (tape->stage_size > 0xffff) {
printk(KERN_NOTICE "ide-tape: decreasing stage size\n"); printk(KERN_NOTICE "ide-tape: decreasing stage size\n");
*ctl /= 2; *ctl /= 2;
tape->stage_size = *ctl * tape->tape_block_size; tape->stage_size = *ctl * tape->blk_size;
} }
stage_size = tape->stage_size; stage_size = tape->stage_size;
tape->pages_per_stage = stage_size / PAGE_SIZE; tape->pages_per_stage = stage_size / PAGE_SIZE;
...@@ -3722,17 +3568,22 @@ static void idetape_setup (ide_drive_t *drive, idetape_tape_t *tape, int minor) ...@@ -3722,17 +3568,22 @@ static void idetape_setup (ide_drive_t *drive, idetape_tape_t *tape, int minor)
tape->max_stages = speed * 1000 * 10 / tape->stage_size; tape->max_stages = speed * 1000 * 10 / tape->stage_size;
/* /* Limit memory use for pipeline to 10% of physical memory */
* Limit memory use for pipeline to 10% of physical memory
*/
si_meminfo(&si); si_meminfo(&si);
if (tape->max_stages * tape->stage_size > si.totalram * si.mem_unit / 10) if (tape->max_stages * tape->stage_size >
tape->max_stages = si.totalram * si.mem_unit / (10 * tape->stage_size); si.totalram * si.mem_unit / 10)
tape->max_stages =
si.totalram * si.mem_unit / (10 * tape->stage_size);
tape->max_stages = min(tape->max_stages, IDETAPE_MAX_PIPELINE_STAGES); tape->max_stages = min(tape->max_stages, IDETAPE_MAX_PIPELINE_STAGES);
tape->min_pipeline = min(tape->max_stages, IDETAPE_MIN_PIPELINE_STAGES); tape->min_pipeline = min(tape->max_stages, IDETAPE_MIN_PIPELINE_STAGES);
tape->max_pipeline = min(tape->max_stages * 2, IDETAPE_MAX_PIPELINE_STAGES); tape->max_pipeline =
if (tape->max_stages == 0) min(tape->max_stages * 2, IDETAPE_MAX_PIPELINE_STAGES);
tape->max_stages = tape->min_pipeline = tape->max_pipeline = 1; if (tape->max_stages == 0) {
tape->max_stages = 1;
tape->min_pipeline = 1;
tape->max_pipeline = 1;
}
t1 = (tape->stage_size * HZ) / (speed * 1000); t1 = (tape->stage_size * HZ) / (speed * 1000);
tmid = (*(u16 *)&tape->caps[16] * 32 * HZ) / (speed * 125); tmid = (*(u16 *)&tape->caps[16] * 32 * HZ) / (speed * 125);
...@@ -3744,17 +3595,19 @@ static void idetape_setup (ide_drive_t *drive, idetape_tape_t *tape, int minor) ...@@ -3744,17 +3595,19 @@ static void idetape_setup (ide_drive_t *drive, idetape_tape_t *tape, int minor)
t = t1; t = t1;
/* /*
* Ensure that the number we got makes sense; limit * Ensure that the number we got makes sense; limit it within
* it within IDETAPE_DSC_RW_MIN and IDETAPE_DSC_RW_MAX. * IDETAPE_DSC_RW_MIN and IDETAPE_DSC_RW_MAX.
*/ */
tape->best_dsc_rw_frequency = max_t(unsigned long, min_t(unsigned long, t, IDETAPE_DSC_RW_MAX), IDETAPE_DSC_RW_MIN); tape->best_dsc_rw_freq = max_t(unsigned long,
min_t(unsigned long, t, IDETAPE_DSC_RW_MAX),
IDETAPE_DSC_RW_MIN);
printk(KERN_INFO "ide-tape: %s <-> %s: %dKBps, %d*%dkB buffer, " printk(KERN_INFO "ide-tape: %s <-> %s: %dKBps, %d*%dkB buffer, "
"%dkB pipeline, %lums tDSC%s\n", "%dkB pipeline, %lums tDSC%s\n",
drive->name, tape->name, *(u16 *)&tape->caps[14], drive->name, tape->name, *(u16 *)&tape->caps[14],
(*(u16 *)&tape->caps[16] * 512) / tape->stage_size, (*(u16 *)&tape->caps[16] * 512) / tape->stage_size,
tape->stage_size / 1024, tape->stage_size / 1024,
tape->max_stages * tape->stage_size / 1024, tape->max_stages * tape->stage_size / 1024,
tape->best_dsc_rw_frequency * 1000 / HZ, tape->best_dsc_rw_freq * 1000 / HZ,
drive->using_dma ? ", DMA":""); drive->using_dma ? ", DMA":"");
idetape_add_settings(drive); idetape_add_settings(drive);
...@@ -3782,7 +3635,8 @@ static void ide_tape_release(struct kref *kref) ...@@ -3782,7 +3635,8 @@ static void ide_tape_release(struct kref *kref)
drive->dsc_overlap = 0; drive->dsc_overlap = 0;
drive->driver_data = NULL; drive->driver_data = NULL;
device_destroy(idetape_sysfs_class, MKDEV(IDETAPE_MAJOR, tape->minor)); device_destroy(idetape_sysfs_class, MKDEV(IDETAPE_MAJOR, tape->minor));
device_destroy(idetape_sysfs_class, MKDEV(IDETAPE_MAJOR, tape->minor + 128)); device_destroy(idetape_sysfs_class,
MKDEV(IDETAPE_MAJOR, tape->minor + 128));
idetape_devs[tape->minor] = NULL; idetape_devs[tape->minor] = NULL;
g->private_data = NULL; g->private_data = NULL;
put_disk(g); put_disk(g);
...@@ -3831,9 +3685,7 @@ static ide_driver_t idetape_driver = { ...@@ -3831,9 +3685,7 @@ static ide_driver_t idetape_driver = {
#endif #endif
}; };
/* /* Our character device supporting functions, passed to register_chrdev. */
* Our character device supporting functions, passed to register_chrdev.
*/
static const struct file_operations idetape_fops = { static const struct file_operations idetape_fops = {
.owner = THIS_MODULE, .owner = THIS_MODULE,
.read = idetape_chrdev_read, .read = idetape_chrdev_read,
...@@ -3848,7 +3700,8 @@ static int idetape_open(struct inode *inode, struct file *filp) ...@@ -3848,7 +3700,8 @@ static int idetape_open(struct inode *inode, struct file *filp)
struct gendisk *disk = inode->i_bdev->bd_disk; struct gendisk *disk = inode->i_bdev->bd_disk;
struct ide_tape_obj *tape; struct ide_tape_obj *tape;
if (!(tape = ide_tape_get(disk))) tape = ide_tape_get(disk);
if (!tape)
return -ENXIO; return -ENXIO;
return 0; return 0;
...@@ -3895,21 +3748,20 @@ static int ide_tape_probe(ide_drive_t *drive) ...@@ -3895,21 +3748,20 @@ static int ide_tape_probe(ide_drive_t *drive)
goto failed; goto failed;
if (drive->media != ide_tape) if (drive->media != ide_tape)
goto failed; goto failed;
if (!idetape_identify_device (drive)) { if (!idetape_identify_device(drive)) {
printk(KERN_ERR "ide-tape: %s: not supported by this version of ide-tape\n", drive->name); printk(KERN_ERR "ide-tape: %s: not supported by this version of"
" the driver\n", drive->name);
goto failed; goto failed;
} }
if (drive->scsi) { if (drive->scsi) {
printk("ide-tape: passing drive %s to ide-scsi emulation.\n", drive->name); printk(KERN_INFO "ide-tape: passing drive %s to ide-scsi"
" emulation.\n", drive->name);
goto failed; goto failed;
} }
if (strstr(drive->id->model, "OnStream DI-")) { tape = kzalloc(sizeof(idetape_tape_t), GFP_KERNEL);
printk(KERN_WARNING "ide-tape: Use drive %s with ide-scsi emulation and osst.\n", drive->name);
printk(KERN_WARNING "ide-tape: OnStream support will be removed soon from ide-tape!\n");
}
tape = kzalloc(sizeof (idetape_tape_t), GFP_KERNEL);
if (tape == NULL) { if (tape == NULL) {
printk(KERN_ERR "ide-tape: %s: Can't allocate a tape structure\n", drive->name); printk(KERN_ERR "ide-tape: %s: Can't allocate a tape struct\n",
drive->name);
goto failed; goto failed;
} }
...@@ -3955,10 +3807,7 @@ static int ide_tape_probe(ide_drive_t *drive) ...@@ -3955,10 +3807,7 @@ static int ide_tape_probe(ide_drive_t *drive)
return -ENODEV; return -ENODEV;
} }
MODULE_DESCRIPTION("ATAPI Streaming TAPE Driver"); static void __exit idetape_exit(void)
MODULE_LICENSE("GPL");
static void __exit idetape_exit (void)
{ {
driver_unregister(&idetape_driver.gen_driver); driver_unregister(&idetape_driver.gen_driver);
class_destroy(idetape_sysfs_class); class_destroy(idetape_sysfs_class);
...@@ -3977,7 +3826,8 @@ static int __init idetape_init(void) ...@@ -3977,7 +3826,8 @@ static int __init idetape_init(void)
} }
if (register_chrdev(IDETAPE_MAJOR, "ht", &idetape_fops)) { if (register_chrdev(IDETAPE_MAJOR, "ht", &idetape_fops)) {
printk(KERN_ERR "ide-tape: Failed to register character device interface\n"); printk(KERN_ERR "ide-tape: Failed to register chrdev"
" interface\n");
error = -EBUSY; error = -EBUSY;
goto out_free_class; goto out_free_class;
} }
...@@ -4000,3 +3850,5 @@ MODULE_ALIAS("ide:*m-tape*"); ...@@ -4000,3 +3850,5 @@ MODULE_ALIAS("ide:*m-tape*");
module_init(idetape_init); module_init(idetape_init);
module_exit(idetape_exit); module_exit(idetape_exit);
MODULE_ALIAS_CHARDEV_MAJOR(IDETAPE_MAJOR); MODULE_ALIAS_CHARDEV_MAJOR(IDETAPE_MAJOR);
MODULE_DESCRIPTION("ATAPI Streaming TAPE Driver");
MODULE_LICENSE("GPL");
...@@ -189,12 +189,11 @@ EXPORT_SYMBOL_GPL(do_rw_taskfile); ...@@ -189,12 +189,11 @@ EXPORT_SYMBOL_GPL(do_rw_taskfile);
*/ */
static ide_startstop_t set_multmode_intr(ide_drive_t *drive) static ide_startstop_t set_multmode_intr(ide_drive_t *drive)
{ {
ide_hwif_t *hwif = HWIF(drive); u8 stat = ide_read_status(drive);
u8 stat;
if (OK_STAT(stat = hwif->INB(IDE_STATUS_REG),READY_STAT,BAD_STAT)) { if (OK_STAT(stat, READY_STAT, BAD_STAT))
drive->mult_count = drive->mult_req; drive->mult_count = drive->mult_req;
} else { else {
drive->mult_req = drive->mult_count = 0; drive->mult_req = drive->mult_count = 0;
drive->special.b.recalibrate = 1; drive->special.b.recalibrate = 1;
(void) ide_dump_status(drive, "set_multmode", stat); (void) ide_dump_status(drive, "set_multmode", stat);
...@@ -207,11 +206,10 @@ static ide_startstop_t set_multmode_intr(ide_drive_t *drive) ...@@ -207,11 +206,10 @@ static ide_startstop_t set_multmode_intr(ide_drive_t *drive)
*/ */
static ide_startstop_t set_geometry_intr(ide_drive_t *drive) static ide_startstop_t set_geometry_intr(ide_drive_t *drive)
{ {
ide_hwif_t *hwif = HWIF(drive);
int retries = 5; int retries = 5;
u8 stat; u8 stat;
while (((stat = hwif->INB(IDE_STATUS_REG)) & BUSY_STAT) && retries--) while (((stat = ide_read_status(drive)) & BUSY_STAT) && retries--)
udelay(10); udelay(10);
if (OK_STAT(stat, READY_STAT, BAD_STAT)) if (OK_STAT(stat, READY_STAT, BAD_STAT))
...@@ -230,10 +228,9 @@ static ide_startstop_t set_geometry_intr(ide_drive_t *drive) ...@@ -230,10 +228,9 @@ static ide_startstop_t set_geometry_intr(ide_drive_t *drive)
*/ */
static ide_startstop_t recal_intr(ide_drive_t *drive) static ide_startstop_t recal_intr(ide_drive_t *drive)
{ {
ide_hwif_t *hwif = HWIF(drive); u8 stat = ide_read_status(drive);
u8 stat;
if (!OK_STAT(stat = hwif->INB(IDE_STATUS_REG), READY_STAT, BAD_STAT)) if (!OK_STAT(stat, READY_STAT, BAD_STAT))
return ide_error(drive, "recal_intr", stat); return ide_error(drive, "recal_intr", stat);
return ide_stopped; return ide_stopped;
} }
...@@ -244,23 +241,23 @@ static ide_startstop_t recal_intr(ide_drive_t *drive) ...@@ -244,23 +241,23 @@ static ide_startstop_t recal_intr(ide_drive_t *drive)
static ide_startstop_t task_no_data_intr(ide_drive_t *drive) static ide_startstop_t task_no_data_intr(ide_drive_t *drive)
{ {
ide_task_t *args = HWGROUP(drive)->rq->special; ide_task_t *args = HWGROUP(drive)->rq->special;
ide_hwif_t *hwif = HWIF(drive);
u8 stat; u8 stat;
local_irq_enable_in_hardirq(); local_irq_enable_in_hardirq();
if (!OK_STAT(stat = hwif->INB(IDE_STATUS_REG),READY_STAT,BAD_STAT)) { stat = ide_read_status(drive);
if (!OK_STAT(stat, READY_STAT, BAD_STAT))
return ide_error(drive, "task_no_data_intr", stat); return ide_error(drive, "task_no_data_intr", stat);
/* calls ide_end_drive_cmd */ /* calls ide_end_drive_cmd */
}
if (args) if (args)
ide_end_drive_cmd(drive, stat, hwif->INB(IDE_ERROR_REG)); ide_end_drive_cmd(drive, stat, ide_read_error(drive));
return ide_stopped; return ide_stopped;
} }
static u8 wait_drive_not_busy(ide_drive_t *drive) static u8 wait_drive_not_busy(ide_drive_t *drive)
{ {
ide_hwif_t *hwif = HWIF(drive);
int retries; int retries;
u8 stat; u8 stat;
...@@ -269,7 +266,9 @@ static u8 wait_drive_not_busy(ide_drive_t *drive) ...@@ -269,7 +266,9 @@ static u8 wait_drive_not_busy(ide_drive_t *drive)
* This can take up to 10 usec, but we will wait max 1 ms. * This can take up to 10 usec, but we will wait max 1 ms.
*/ */
for (retries = 0; retries < 100; retries++) { for (retries = 0; retries < 100; retries++) {
if ((stat = hwif->INB(IDE_STATUS_REG)) & BUSY_STAT) stat = ide_read_status(drive);
if (stat & BUSY_STAT)
udelay(10); udelay(10);
else else
break; break;
...@@ -408,7 +407,7 @@ static ide_startstop_t task_error(ide_drive_t *drive, struct request *rq, ...@@ -408,7 +407,7 @@ static ide_startstop_t task_error(ide_drive_t *drive, struct request *rq,
void task_end_request(ide_drive_t *drive, struct request *rq, u8 stat) void task_end_request(ide_drive_t *drive, struct request *rq, u8 stat)
{ {
if (rq->cmd_type == REQ_TYPE_ATA_TASKFILE) { if (rq->cmd_type == REQ_TYPE_ATA_TASKFILE) {
u8 err = drive->hwif->INB(IDE_ERROR_REG); u8 err = ide_read_error(drive);
ide_end_drive_cmd(drive, stat, err); ide_end_drive_cmd(drive, stat, err);
return; return;
...@@ -430,7 +429,7 @@ static ide_startstop_t task_in_intr(ide_drive_t *drive) ...@@ -430,7 +429,7 @@ static ide_startstop_t task_in_intr(ide_drive_t *drive)
{ {
ide_hwif_t *hwif = drive->hwif; ide_hwif_t *hwif = drive->hwif;
struct request *rq = HWGROUP(drive)->rq; struct request *rq = HWGROUP(drive)->rq;
u8 stat = hwif->INB(IDE_STATUS_REG); u8 stat = ide_read_status(drive);
/* new way for dealing with premature shared PCI interrupts */ /* new way for dealing with premature shared PCI interrupts */
if (!OK_STAT(stat, DRQ_STAT, BAD_R_STAT)) { if (!OK_STAT(stat, DRQ_STAT, BAD_R_STAT)) {
...@@ -465,7 +464,7 @@ static ide_startstop_t task_out_intr (ide_drive_t *drive) ...@@ -465,7 +464,7 @@ static ide_startstop_t task_out_intr (ide_drive_t *drive)
{ {
ide_hwif_t *hwif = drive->hwif; ide_hwif_t *hwif = drive->hwif;
struct request *rq = HWGROUP(drive)->rq; struct request *rq = HWGROUP(drive)->rq;
u8 stat = hwif->INB(IDE_STATUS_REG); u8 stat = ide_read_status(drive);
if (!OK_STAT(stat, DRIVE_READY, drive->bad_wstat)) if (!OK_STAT(stat, DRIVE_READY, drive->bad_wstat))
return task_error(drive, rq, __FUNCTION__, stat); return task_error(drive, rq, __FUNCTION__, stat);
......
...@@ -618,60 +618,6 @@ void ide_unregister(unsigned int index, int init_default, int restore) ...@@ -618,60 +618,6 @@ void ide_unregister(unsigned int index, int init_default, int restore)
EXPORT_SYMBOL(ide_unregister); EXPORT_SYMBOL(ide_unregister);
/**
* ide_setup_ports - set up IDE interface ports
* @hw: register descriptions
* @base: base register
* @offsets: table of register offsets
* @ctrl: control register
* @ack_irq: IRQ ack
* @irq: interrupt lie
*
* Setup hw_regs_t structure described by parameters. You
* may set up the hw structure yourself OR use this routine to
* do it for you. This is basically a helper
*
*/
void ide_setup_ports ( hw_regs_t *hw,
unsigned long base, int *offsets,
unsigned long ctrl, unsigned long intr,
ide_ack_intr_t *ack_intr,
/*
* ide_io_ops_t *iops,
*/
int irq)
{
int i;
memset(hw, 0, sizeof(hw_regs_t));
for (i = 0; i < IDE_NR_PORTS; i++) {
if (offsets[i] == -1) {
switch(i) {
case IDE_CONTROL_OFFSET:
hw->io_ports[i] = ctrl;
break;
#if defined(CONFIG_AMIGA) || defined(CONFIG_MAC)
case IDE_IRQ_OFFSET:
hw->io_ports[i] = intr;
break;
#endif /* (CONFIG_AMIGA) || (CONFIG_MAC) */
default:
hw->io_ports[i] = 0;
break;
}
} else {
hw->io_ports[i] = base + offsets[i];
}
}
hw->irq = irq;
hw->ack_intr = ack_intr;
/*
* hw->iops = iops;
*/
}
void ide_init_port_hw(ide_hwif_t *hwif, hw_regs_t *hw) void ide_init_port_hw(ide_hwif_t *hwif, hw_regs_t *hw)
{ {
memcpy(hwif->io_ports, hw->io_ports, sizeof(hwif->io_ports)); memcpy(hwif->io_ports, hw->io_ports, sizeof(hwif->io_ports));
......
...@@ -56,31 +56,11 @@ static u_int xsurf_bases[XSURF_NUM_HWIFS] __initdata = { ...@@ -56,31 +56,11 @@ static u_int xsurf_bases[XSURF_NUM_HWIFS] __initdata = {
XSURF_BASE1, XSURF_BASE2 XSURF_BASE1, XSURF_BASE2
}; };
/* /*
* Offsets from one of the above bases * Offsets from one of the above bases
*/ */
#define BUDDHA_DATA 0x00
#define BUDDHA_ERROR 0x06 /* see err-bits */
#define BUDDHA_NSECTOR 0x0a /* nr of sectors to read/write */
#define BUDDHA_SECTOR 0x0e /* starting sector */
#define BUDDHA_LCYL 0x12 /* starting cylinder */
#define BUDDHA_HCYL 0x16 /* high byte of starting cyl */
#define BUDDHA_SELECT 0x1a /* 101dhhhh , d=drive, hhhh=head */
#define BUDDHA_STATUS 0x1e /* see status-bits */
#define BUDDHA_CONTROL 0x11a #define BUDDHA_CONTROL 0x11a
#define XSURF_CONTROL -1 /* X-Surf has no CS1* (Control/AltStat) */
static int buddha_offsets[IDE_NR_PORTS] __initdata = {
BUDDHA_DATA, BUDDHA_ERROR, BUDDHA_NSECTOR, BUDDHA_SECTOR, BUDDHA_LCYL,
BUDDHA_HCYL, BUDDHA_SELECT, BUDDHA_STATUS, BUDDHA_CONTROL, -1
};
static int xsurf_offsets[IDE_NR_PORTS] __initdata = {
BUDDHA_DATA, BUDDHA_ERROR, BUDDHA_NSECTOR, BUDDHA_SECTOR, BUDDHA_LCYL,
BUDDHA_HCYL, BUDDHA_SELECT, BUDDHA_STATUS, XSURF_CONTROL, -1
};
/* /*
* Other registers * Other registers
...@@ -140,6 +120,26 @@ static int xsurf_ack_intr(ide_hwif_t *hwif) ...@@ -140,6 +120,26 @@ static int xsurf_ack_intr(ide_hwif_t *hwif)
return 1; return 1;
} }
static void __init buddha_setup_ports(hw_regs_t *hw, unsigned long base,
unsigned long ctl, unsigned long irq_port,
ide_ack_intr_t *ack_intr)
{
int i;
memset(hw, 0, sizeof(*hw));
hw->io_ports[IDE_DATA_OFFSET] = base;
for (i = 1; i < 8; i++)
hw->io_ports[i] = base + 2 + i * 4;
hw->io_ports[IDE_CONTROL_OFFSET] = ctl;
hw->io_ports[IDE_IRQ_OFFSET] = irq_port;
hw->irq = IRQ_AMIGA_PORTS;
hw->ack_intr = ack_intr;
}
/* /*
* Probe for a Buddha or Catweasel IDE interface * Probe for a Buddha or Catweasel IDE interface
*/ */
...@@ -202,22 +202,24 @@ static int __init buddha_init(void) ...@@ -202,22 +202,24 @@ static int __init buddha_init(void)
printk(KERN_INFO "ide: %s IDE controller\n", printk(KERN_INFO "ide: %s IDE controller\n",
buddha_board_name[type]); buddha_board_name[type]);
for(i=0;i<buddha_num_hwifs;i++) { for (i = 0; i < buddha_num_hwifs; i++) {
if(type != BOARD_XSURF) { unsigned long base, ctl, irq_port;
ide_setup_ports(&hw, (buddha_board+buddha_bases[i]), ide_ack_intr_t *ack_intr;
buddha_offsets, 0,
(buddha_board+buddha_irqports[i]), if (type != BOARD_XSURF) {
buddha_ack_intr, base = buddha_board + buddha_bases[i];
// budda_iops, ctl = base + BUDDHA_CONTROL;
IRQ_AMIGA_PORTS); irq_port = buddha_board + buddha_irqports[i];
ack_intr = buddha_ack_intr;
} else { } else {
ide_setup_ports(&hw, (buddha_board+xsurf_bases[i]), base = buddha_board + xsurf_bases[i];
xsurf_offsets, 0, /* X-Surf has no CS1* (Control/AltStat) */
(buddha_board+xsurf_irqports[i]), ctl = 0;
xsurf_ack_intr, irq_port = buddha_board + xsurf_irqports[i];
// xsurf_iops, ack_intr = xsurf_ack_intr;
IRQ_AMIGA_PORTS); }
}
buddha_setup_ports(&hw, base, ctl, irq_port, ack_intr);
hwif = ide_find_port(hw.io_ports[IDE_DATA_OFFSET]); hwif = ide_find_port(hw.io_ports[IDE_DATA_OFFSET]);
if (hwif) { if (hwif) {
......
...@@ -33,22 +33,8 @@ ...@@ -33,22 +33,8 @@
* Offsets from the above base * Offsets from the above base
*/ */
#define ATA_HD_DATA 0x00
#define ATA_HD_ERROR 0x05 /* see err-bits */
#define ATA_HD_NSECTOR 0x09 /* nr of sectors to read/write */
#define ATA_HD_SECTOR 0x0d /* starting sector */
#define ATA_HD_LCYL 0x11 /* starting cylinder */
#define ATA_HD_HCYL 0x15 /* high byte of starting cyl */
#define ATA_HD_SELECT 0x19 /* 101dhhhh , d=drive, hhhh=head */
#define ATA_HD_STATUS 0x1d /* see status-bits */
#define ATA_HD_CONTROL 0x39 #define ATA_HD_CONTROL 0x39
static int falconide_offsets[IDE_NR_PORTS] __initdata = {
ATA_HD_DATA, ATA_HD_ERROR, ATA_HD_NSECTOR, ATA_HD_SECTOR, ATA_HD_LCYL,
ATA_HD_HCYL, ATA_HD_SELECT, ATA_HD_STATUS, ATA_HD_CONTROL, -1
};
/* /*
* falconide_intr_lock is used to obtain access to the IDE interrupt, * falconide_intr_lock is used to obtain access to the IDE interrupt,
* which is shared between several drivers. * which is shared between several drivers.
...@@ -57,6 +43,22 @@ static int falconide_offsets[IDE_NR_PORTS] __initdata = { ...@@ -57,6 +43,22 @@ static int falconide_offsets[IDE_NR_PORTS] __initdata = {
int falconide_intr_lock; int falconide_intr_lock;
EXPORT_SYMBOL(falconide_intr_lock); EXPORT_SYMBOL(falconide_intr_lock);
static void __init falconide_setup_ports(hw_regs_t *hw)
{
int i;
memset(hw, 0, sizeof(*hw));
hw->io_ports[IDE_DATA_OFFSET] = ATA_HD_BASE;
for (i = 1; i < 8; i++)
hw->io_ports[i] = ATA_HD_BASE + 1 + i * 4;
hw->io_ports[IDE_CONTROL_OFFSET] = ATA_HD_CONTROL;
hw->irq = IRQ_MFP_IDE;
hw->ack_intr = NULL;
}
/* /*
* Probe for a Falcon IDE interface * Probe for a Falcon IDE interface
...@@ -64,16 +66,15 @@ EXPORT_SYMBOL(falconide_intr_lock); ...@@ -64,16 +66,15 @@ EXPORT_SYMBOL(falconide_intr_lock);
static int __init falconide_init(void) static int __init falconide_init(void)
{ {
if (MACH_IS_ATARI && ATARIHW_PRESENT(IDE)) {
hw_regs_t hw; hw_regs_t hw;
ide_hwif_t *hwif; ide_hwif_t *hwif;
if (!MACH_IS_ATARI || !ATARIHW_PRESENT(IDE))
return 0;
printk(KERN_INFO "ide: Falcon IDE controller\n"); printk(KERN_INFO "ide: Falcon IDE controller\n");
ide_setup_ports(&hw, ATA_HD_BASE, falconide_offsets, falconide_setup_ports(&hw);
0, 0, NULL,
// falconide_iops,
IRQ_MFP_IDE);
hwif = ide_find_port(hw.io_ports[IDE_DATA_OFFSET]); hwif = ide_find_port(hw.io_ports[IDE_DATA_OFFSET]);
if (hwif) { if (hwif) {
...@@ -85,9 +86,8 @@ static int __init falconide_init(void) ...@@ -85,9 +86,8 @@ static int __init falconide_init(void)
ide_device_add(idx, NULL); ide_device_add(idx, NULL);
} }
}
return 0; return 0;
} }
module_init(falconide_init); module_init(falconide_init);
...@@ -34,22 +34,8 @@ ...@@ -34,22 +34,8 @@
* Offsets from one of the above bases * Offsets from one of the above bases
*/ */
#define GAYLE_DATA 0x00
#define GAYLE_ERROR 0x06 /* see err-bits */
#define GAYLE_NSECTOR 0x0a /* nr of sectors to read/write */
#define GAYLE_SECTOR 0x0e /* starting sector */
#define GAYLE_LCYL 0x12 /* starting cylinder */
#define GAYLE_HCYL 0x16 /* high byte of starting cyl */
#define GAYLE_SELECT 0x1a /* 101dhhhh , d=drive, hhhh=head */
#define GAYLE_STATUS 0x1e /* see status-bits */
#define GAYLE_CONTROL 0x101a #define GAYLE_CONTROL 0x101a
static int gayle_offsets[IDE_NR_PORTS] __initdata = {
GAYLE_DATA, GAYLE_ERROR, GAYLE_NSECTOR, GAYLE_SECTOR, GAYLE_LCYL,
GAYLE_HCYL, GAYLE_SELECT, GAYLE_STATUS, -1, -1
};
/* /*
* These are at different offsets from the base * These are at different offsets from the base
*/ */
...@@ -106,6 +92,26 @@ static int gayle_ack_intr_a1200(ide_hwif_t *hwif) ...@@ -106,6 +92,26 @@ static int gayle_ack_intr_a1200(ide_hwif_t *hwif)
return 1; return 1;
} }
static void __init gayle_setup_ports(hw_regs_t *hw, unsigned long base,
unsigned long ctl, unsigned long irq_port,
ide_ack_intr_t *ack_intr);
{
int i;
memset(hw, 0, sizeof(*hw));
hw->io_ports[IDE_DATA_OFFSET] = base;
for (i = 1; i < 8; i++)
hw->io_ports[i] = base + 2 + i * 4;
hw->io_ports[IDE_CONTROL_OFFSET] = ctl;
hw->io_ports[IDE_IRQ_OFFSET] = irq_port;
hw->irq = IRQ_AMIGA_PORTS;
hw->ack_intr = ack_intr;
}
/* /*
* Probe for a Gayle IDE interface (and optionally for an IDE doubler) * Probe for a Gayle IDE interface (and optionally for an IDE doubler)
*/ */
...@@ -167,10 +173,7 @@ static int __init gayle_init(void) ...@@ -167,10 +173,7 @@ static int __init gayle_init(void)
base = (unsigned long)ZTWO_VADDR(phys_base); base = (unsigned long)ZTWO_VADDR(phys_base);
ctrlport = GAYLE_HAS_CONTROL_REG ? (base + GAYLE_CONTROL) : 0; ctrlport = GAYLE_HAS_CONTROL_REG ? (base + GAYLE_CONTROL) : 0;
ide_setup_ports(&hw, base, gayle_offsets, gayle_setup_ports(&hw, base, ctrlport, irqport, ack_intr);
ctrlport, irqport, ack_intr,
// &gayle_iops,
IRQ_AMIGA_PORTS);
hwif = ide_find_port(base); hwif = ide_find_port(base);
if (hwif) { if (hwif) {
......
...@@ -421,11 +421,14 @@ static void bad_rw_intr(void) ...@@ -421,11 +421,14 @@ static void bad_rw_intr(void)
static inline int wait_DRQ(void) static inline int wait_DRQ(void)
{ {
int retries = 100000, stat; int retries;
int stat;
while (--retries > 0) for (retries = 0; retries < 100000; retries++) {
if ((stat = inb_p(HD_STATUS)) & DRQ_STAT) stat = inb_p(HD_STATUS);
if (stat & DRQ_STAT)
return 0; return 0;
}
dump_status("wait_DRQ", stat); dump_status("wait_DRQ", stat);
return -1; return -1;
} }
......
...@@ -31,14 +31,6 @@ ...@@ -31,14 +31,6 @@
* These match MkLinux so they should be correct. * These match MkLinux so they should be correct.
*/ */
#define IDE_DATA 0x00
#define IDE_ERROR 0x04 /* see err-bits */
#define IDE_NSECTOR 0x08 /* nr of sectors to read/write */
#define IDE_SECTOR 0x0c /* starting sector */
#define IDE_LCYL 0x10 /* starting cylinder */
#define IDE_HCYL 0x14 /* high byte of starting cyl */
#define IDE_SELECT 0x18 /* 101dhhhh , d=drive, hhhh=head */
#define IDE_STATUS 0x1c /* see status-bits */
#define IDE_CONTROL 0x38 /* control/altstatus */ #define IDE_CONTROL 0x38 /* control/altstatus */
/* /*
...@@ -63,11 +55,6 @@ ...@@ -63,11 +55,6 @@
volatile unsigned char *ide_ifr = (unsigned char *) (IDE_BASE + IDE_IFR); volatile unsigned char *ide_ifr = (unsigned char *) (IDE_BASE + IDE_IFR);
static int macide_offsets[IDE_NR_PORTS] = {
IDE_DATA, IDE_ERROR, IDE_NSECTOR, IDE_SECTOR, IDE_LCYL,
IDE_HCYL, IDE_SELECT, IDE_STATUS, IDE_CONTROL
};
int macide_ack_intr(ide_hwif_t* hwif) int macide_ack_intr(ide_hwif_t* hwif)
{ {
if (*ide_ifr & 0x20) { if (*ide_ifr & 0x20) {
...@@ -77,6 +64,22 @@ int macide_ack_intr(ide_hwif_t* hwif) ...@@ -77,6 +64,22 @@ int macide_ack_intr(ide_hwif_t* hwif)
return 0; return 0;
} }
static void __init macide_setup_ports(hw_regs_t *hw, unsigned long base,
int irq, ide_ack_intr_t *ack_intr)
{
int i;
memset(hw, 0, sizeof(*hw));
for (i = 0; i < 8; i++)
hw->io_ports[i] = base + i * 4;
hw->io_ports[IDE_CONTROL_OFFSET] = IDE_CONTROL;
hw->irq = irq;
hw->ack_intr = ack_intr;
}
static const char *mac_ide_name[] = static const char *mac_ide_name[] =
{ "Quadra", "Powerbook", "Powerbook Baboon" }; { "Quadra", "Powerbook", "Powerbook Baboon" };
...@@ -86,27 +89,27 @@ static const char *mac_ide_name[] = ...@@ -86,27 +89,27 @@ static const char *mac_ide_name[] =
static int __init macide_init(void) static int __init macide_init(void)
{ {
hw_regs_t hw;
ide_hwif_t *hwif; ide_hwif_t *hwif;
ide_ack_intr_t *ack_intr;
unsigned long base;
int irq;
hw_regs_t hw;
switch (macintosh_config->ide_type) { switch (macintosh_config->ide_type) {
case MAC_IDE_QUADRA: case MAC_IDE_QUADRA:
ide_setup_ports(&hw, IDE_BASE, macide_offsets, base = IDE_BASE;
0, 0, macide_ack_intr, ack_intr = macide_ack_intr;
// quadra_ide_iops, irq = IRQ_NUBUS_F;
IRQ_NUBUS_F);
break; break;
case MAC_IDE_PB: case MAC_IDE_PB:
ide_setup_ports(&hw, IDE_BASE, macide_offsets, base = IDE_BASE;
0, 0, macide_ack_intr, ack_intr = macide_ack_intr;
// macide_pb_iops, irq = IRQ_NUBUS_C;
IRQ_NUBUS_C);
break; break;
case MAC_IDE_BABOON: case MAC_IDE_BABOON:
ide_setup_ports(&hw, BABOON_BASE, macide_offsets, base = BABOON_BASE;
0, 0, NULL, ack_intr = NULL;
// macide_baboon_iops, irq = IRQ_BABOON_1;
IRQ_BABOON_1);
break; break;
default: default:
return -ENODEV; return -ENODEV;
...@@ -115,6 +118,8 @@ static int __init macide_init(void) ...@@ -115,6 +118,8 @@ static int __init macide_init(void)
printk(KERN_INFO "ide: Macintosh %s IDE controller\n", printk(KERN_INFO "ide: Macintosh %s IDE controller\n",
mac_ide_name[macintosh_config->ide_type - 1]); mac_ide_name[macintosh_config->ide_type - 1]);
macide_setup_ports(&hw, base, irq, ack_intr);
hwif = ide_find_port(hw.io_ports[IDE_DATA_OFFSET]); hwif = ide_find_port(hw.io_ports[IDE_DATA_OFFSET]);
if (hwif) { if (hwif) {
u8 index = hwif->index; u8 index = hwif->index;
......
...@@ -66,16 +66,12 @@ static int q40ide_default_irq(unsigned long base) ...@@ -66,16 +66,12 @@ static int q40ide_default_irq(unsigned long base)
/* /*
* This is very similar to ide_setup_ports except that addresses * Addresses are pretranslated for Q40 ISA access.
* are pretranslated for q40 ISA access
*/ */
void q40_ide_setup_ports ( hw_regs_t *hw, void q40_ide_setup_ports ( hw_regs_t *hw,
unsigned long base, int *offsets, unsigned long base, int *offsets,
unsigned long ctrl, unsigned long intr, unsigned long ctrl, unsigned long intr,
ide_ack_intr_t *ack_intr, ide_ack_intr_t *ack_intr,
/*
* ide_io_ops_t *iops,
*/
int irq) int irq)
{ {
int i; int i;
...@@ -92,9 +88,6 @@ void q40_ide_setup_ports ( hw_regs_t *hw, ...@@ -92,9 +88,6 @@ void q40_ide_setup_ports ( hw_regs_t *hw,
hw->irq = irq; hw->irq = irq;
hw->ack_intr = ack_intr; hw->ack_intr = ack_intr;
/*
* hw->iops = iops;
*/
} }
......
...@@ -34,7 +34,8 @@ obj-$(CONFIG_BLK_DEV_TRM290) += trm290.o ...@@ -34,7 +34,8 @@ obj-$(CONFIG_BLK_DEV_TRM290) += trm290.o
obj-$(CONFIG_BLK_DEV_VIA82CXXX) += via82cxxx.o obj-$(CONFIG_BLK_DEV_VIA82CXXX) += via82cxxx.o
# Must appear at the end of the block # Must appear at the end of the block
obj-$(CONFIG_BLK_DEV_GENERIC) += generic.o obj-$(CONFIG_BLK_DEV_GENERIC) += ide-pci-generic.o
ide-pci-generic-y += generic.o
ifeq ($(CONFIG_BLK_DEV_CMD640), m) ifeq ($(CONFIG_BLK_DEV_CMD640), m)
obj-m += cmd640.o obj-m += cmd640.o
......
...@@ -29,19 +29,6 @@ ...@@ -29,19 +29,6 @@
static int ide_generic_all; /* Set to claim all devices */ static int ide_generic_all; /* Set to claim all devices */
/*
* the module_param_named() was added for the modular case
* the __setup() is left as compatibility for existing setups
*/
#ifndef MODULE
static int __init ide_generic_all_on(char *unused)
{
ide_generic_all = 1;
printk(KERN_INFO "IDE generic will claim all unknown PCI IDE storage controllers.\n");
return 1;
}
const __setup("all-generic-ide", ide_generic_all_on);
#endif
module_param_named(all_generic_ide, ide_generic_all, bool, 0444); module_param_named(all_generic_ide, ide_generic_all, bool, 0444);
MODULE_PARM_DESC(all_generic_ide, "IDE generic will claim all unknown PCI IDE storage controllers."); MODULE_PARM_DESC(all_generic_ide, "IDE generic will claim all unknown PCI IDE storage controllers.");
......
...@@ -704,9 +704,6 @@ static void __devinit init_mmio_iops_siimage(ide_hwif_t *hwif) ...@@ -704,9 +704,6 @@ static void __devinit init_mmio_iops_siimage(ide_hwif_t *hwif)
hwif->sata_scr[SATA_STATUS_OFFSET] = base + 0x104; hwif->sata_scr[SATA_STATUS_OFFSET] = base + 0x104;
hwif->sata_scr[SATA_ERROR_OFFSET] = base + 0x108; hwif->sata_scr[SATA_ERROR_OFFSET] = base + 0x108;
hwif->sata_scr[SATA_CONTROL_OFFSET] = base + 0x100; hwif->sata_scr[SATA_CONTROL_OFFSET] = base + 0x100;
hwif->sata_misc[SATA_MISC_OFFSET] = base + 0x140;
hwif->sata_misc[SATA_PHY_OFFSET] = base + 0x144;
hwif->sata_misc[SATA_IEN_OFFSET] = base + 0x148;
} }
memcpy(hwif->io_ports, hw.io_ports, sizeof(hwif->io_ports)); memcpy(hwif->io_ports, hw.io_ports, sizeof(hwif->io_ports));
......
...@@ -78,12 +78,14 @@ struct media_bay_info { ...@@ -78,12 +78,14 @@ struct media_bay_info {
int cached_gpio; int cached_gpio;
int sleeping; int sleeping;
struct semaphore lock; struct semaphore lock;
#ifdef CONFIG_BLK_DEV_IDE #ifdef CONFIG_BLK_DEV_IDE_PMAC
void __iomem *cd_base; void __iomem *cd_base;
int cd_index;
int cd_irq; int cd_irq;
int cd_retry; int cd_retry;
#endif #endif
#if defined(CONFIG_BLK_DEV_IDE_PMAC) || defined(CONFIG_MAC_FLOPPY)
int cd_index;
#endif
}; };
#define MAX_BAYS 2 #define MAX_BAYS 2
...@@ -91,7 +93,7 @@ struct media_bay_info { ...@@ -91,7 +93,7 @@ struct media_bay_info {
static struct media_bay_info media_bays[MAX_BAYS]; static struct media_bay_info media_bays[MAX_BAYS];
int media_bay_count = 0; int media_bay_count = 0;
#ifdef CONFIG_BLK_DEV_IDE #ifdef CONFIG_BLK_DEV_IDE_PMAC
/* check the busy bit in the media-bay ide interface /* check the busy bit in the media-bay ide interface
(assumes the media-bay contains an ide device) */ (assumes the media-bay contains an ide device) */
#define MB_IDE_READY(i) ((readb(media_bays[i].cd_base + 0x70) & 0x80) == 0) #define MB_IDE_READY(i) ((readb(media_bays[i].cd_base + 0x70) & 0x80) == 0)
...@@ -401,7 +403,7 @@ static void poll_media_bay(struct media_bay_info* bay) ...@@ -401,7 +403,7 @@ static void poll_media_bay(struct media_bay_info* bay)
set_mb_power(bay, id != MB_NO); set_mb_power(bay, id != MB_NO);
bay->content_id = id; bay->content_id = id;
if (id == MB_NO) { if (id == MB_NO) {
#ifdef CONFIG_BLK_DEV_IDE #ifdef CONFIG_BLK_DEV_IDE_PMAC
bay->cd_retry = 0; bay->cd_retry = 0;
#endif #endif
printk(KERN_INFO "media bay %d is empty\n", bay->index); printk(KERN_INFO "media bay %d is empty\n", bay->index);
...@@ -414,9 +416,9 @@ static void poll_media_bay(struct media_bay_info* bay) ...@@ -414,9 +416,9 @@ static void poll_media_bay(struct media_bay_info* bay)
} }
} }
#ifdef CONFIG_MAC_FLOPPY
int check_media_bay(struct device_node *which_bay, int what) int check_media_bay(struct device_node *which_bay, int what)
{ {
#ifdef CONFIG_BLK_DEV_IDE
int i; int i;
for (i=0; i<media_bay_count; i++) for (i=0; i<media_bay_count; i++)
...@@ -426,14 +428,14 @@ int check_media_bay(struct device_node *which_bay, int what) ...@@ -426,14 +428,14 @@ int check_media_bay(struct device_node *which_bay, int what)
media_bays[i].cd_index = -1; media_bays[i].cd_index = -1;
return -EINVAL; return -EINVAL;
} }
#endif /* CONFIG_BLK_DEV_IDE */
return -ENODEV; return -ENODEV;
} }
EXPORT_SYMBOL(check_media_bay); EXPORT_SYMBOL(check_media_bay);
#endif /* CONFIG_MAC_FLOPPY */
#ifdef CONFIG_BLK_DEV_IDE_PMAC
int check_media_bay_by_base(unsigned long base, int what) int check_media_bay_by_base(unsigned long base, int what)
{ {
#ifdef CONFIG_BLK_DEV_IDE
int i; int i;
for (i=0; i<media_bay_count; i++) for (i=0; i<media_bay_count; i++)
...@@ -443,15 +445,13 @@ int check_media_bay_by_base(unsigned long base, int what) ...@@ -443,15 +445,13 @@ int check_media_bay_by_base(unsigned long base, int what)
media_bays[i].cd_index = -1; media_bays[i].cd_index = -1;
return -EINVAL; return -EINVAL;
} }
#endif
return -ENODEV; return -ENODEV;
} }
int media_bay_set_ide_infos(struct device_node* which_bay, unsigned long base, int media_bay_set_ide_infos(struct device_node* which_bay, unsigned long base,
int irq, int index) int irq, int index)
{ {
#ifdef CONFIG_BLK_DEV_IDE
int i; int i;
for (i=0; i<media_bay_count; i++) { for (i=0; i<media_bay_count; i++) {
...@@ -483,10 +483,10 @@ int media_bay_set_ide_infos(struct device_node* which_bay, unsigned long base, ...@@ -483,10 +483,10 @@ int media_bay_set_ide_infos(struct device_node* which_bay, unsigned long base,
return -ENODEV; return -ENODEV;
} }
} }
#endif /* CONFIG_BLK_DEV_IDE */
return -ENODEV; return -ENODEV;
} }
#endif /* CONFIG_BLK_DEV_IDE_PMAC */
static void media_bay_step(int i) static void media_bay_step(int i)
{ {
...@@ -521,14 +521,13 @@ static void media_bay_step(int i) ...@@ -521,14 +521,13 @@ static void media_bay_step(int i)
bay->state = mb_resetting; bay->state = mb_resetting;
MBDBG("mediabay%d: waiting reset (kind:%d)\n", i, bay->content_id); MBDBG("mediabay%d: waiting reset (kind:%d)\n", i, bay->content_id);
break; break;
case mb_resetting: case mb_resetting:
if (bay->content_id != MB_CD) { if (bay->content_id != MB_CD) {
MBDBG("mediabay%d: bay is up (kind:%d)\n", i, bay->content_id); MBDBG("mediabay%d: bay is up (kind:%d)\n", i, bay->content_id);
bay->state = mb_up; bay->state = mb_up;
break; break;
} }
#ifdef CONFIG_BLK_DEV_IDE #ifdef CONFIG_BLK_DEV_IDE_PMAC
MBDBG("mediabay%d: waiting IDE reset (kind:%d)\n", i, bay->content_id); MBDBG("mediabay%d: waiting IDE reset (kind:%d)\n", i, bay->content_id);
bay->ops->un_reset_ide(bay); bay->ops->un_reset_ide(bay);
bay->timer = msecs_to_jiffies(MB_IDE_WAIT); bay->timer = msecs_to_jiffies(MB_IDE_WAIT);
...@@ -536,16 +535,14 @@ static void media_bay_step(int i) ...@@ -536,16 +535,14 @@ static void media_bay_step(int i)
#else #else
printk(KERN_DEBUG "media-bay %d is ide (not compiled in kernel)\n", i); printk(KERN_DEBUG "media-bay %d is ide (not compiled in kernel)\n", i);
set_mb_power(bay, 0); set_mb_power(bay, 0);
#endif /* CONFIG_BLK_DEV_IDE */ #endif /* CONFIG_BLK_DEV_IDE_PMAC */
break; break;
#ifdef CONFIG_BLK_DEV_IDE_PMAC
#ifdef CONFIG_BLK_DEV_IDE
case mb_ide_resetting: case mb_ide_resetting:
bay->timer = msecs_to_jiffies(MB_IDE_TIMEOUT); bay->timer = msecs_to_jiffies(MB_IDE_TIMEOUT);
bay->state = mb_ide_waiting; bay->state = mb_ide_waiting;
MBDBG("mediabay%d: waiting IDE ready (kind:%d)\n", i, bay->content_id); MBDBG("mediabay%d: waiting IDE ready (kind:%d)\n", i, bay->content_id);
break; break;
case mb_ide_waiting: case mb_ide_waiting:
if (bay->cd_base == NULL) { if (bay->cd_base == NULL) {
bay->timer = 0; bay->timer = 0;
...@@ -587,11 +584,10 @@ static void media_bay_step(int i) ...@@ -587,11 +584,10 @@ static void media_bay_step(int i)
bay->timer = 0; bay->timer = 0;
} }
break; break;
#endif /* CONFIG_BLK_DEV_IDE */ #endif /* CONFIG_BLK_DEV_IDE_PMAC */
case mb_powering_down: case mb_powering_down:
bay->state = mb_empty; bay->state = mb_empty;
#ifdef CONFIG_BLK_DEV_IDE #ifdef CONFIG_BLK_DEV_IDE_PMAC
if (bay->cd_index >= 0) { if (bay->cd_index >= 0) {
printk(KERN_DEBUG "Unregistering mb %d ide, index:%d\n", i, printk(KERN_DEBUG "Unregistering mb %d ide, index:%d\n", i,
bay->cd_index); bay->cd_index);
...@@ -607,7 +603,7 @@ static void media_bay_step(int i) ...@@ -607,7 +603,7 @@ static void media_bay_step(int i)
bay->content_id = MB_NO; bay->content_id = MB_NO;
} }
} }
#endif /* CONFIG_BLK_DEV_IDE */ #endif /* CONFIG_BLK_DEV_IDE_PMAC */
MBDBG("mediabay%d: end of power down\n", i); MBDBG("mediabay%d: end of power down\n", i);
break; break;
} }
...@@ -739,7 +735,7 @@ static int media_bay_resume(struct macio_dev *mdev) ...@@ -739,7 +735,7 @@ static int media_bay_resume(struct macio_dev *mdev)
bay->last_value = bay->content_id; bay->last_value = bay->content_id;
bay->value_count = msecs_to_jiffies(MB_STABLE_DELAY); bay->value_count = msecs_to_jiffies(MB_STABLE_DELAY);
bay->timer = msecs_to_jiffies(MB_POWER_DELAY); bay->timer = msecs_to_jiffies(MB_POWER_DELAY);
#ifdef CONFIG_BLK_DEV_IDE #ifdef CONFIG_BLK_DEV_IDE_PMAC
bay->cd_retry = 0; bay->cd_retry = 0;
#endif #endif
do { do {
...@@ -829,7 +825,7 @@ static int __init media_bay_init(void) ...@@ -829,7 +825,7 @@ static int __init media_bay_init(void)
for (i=0; i<MAX_BAYS; i++) { for (i=0; i<MAX_BAYS; i++) {
memset((char *)&media_bays[i], 0, sizeof(struct media_bay_info)); memset((char *)&media_bays[i], 0, sizeof(struct media_bay_info));
media_bays[i].content_id = -1; media_bays[i].content_id = -1;
#ifdef CONFIG_BLK_DEV_IDE #ifdef CONFIG_BLK_DEV_IDE_PMAC
media_bays[i].cd_index = -1; media_bays[i].cd_index = -1;
#endif #endif
} }
......
...@@ -287,7 +287,7 @@ static int idescsi_end_request(ide_drive_t *, int, int); ...@@ -287,7 +287,7 @@ static int idescsi_end_request(ide_drive_t *, int, int);
static ide_startstop_t static ide_startstop_t
idescsi_atapi_error(ide_drive_t *drive, struct request *rq, u8 stat, u8 err) idescsi_atapi_error(ide_drive_t *drive, struct request *rq, u8 stat, u8 err)
{ {
if (HWIF(drive)->INB(IDE_STATUS_REG) & (BUSY_STAT|DRQ_STAT)) if (ide_read_status(drive) & (BUSY_STAT | DRQ_STAT))
/* force an abort */ /* force an abort */
HWIF(drive)->OUTB(WIN_IDLEIMMEDIATE,IDE_COMMAND_REG); HWIF(drive)->OUTB(WIN_IDLEIMMEDIATE,IDE_COMMAND_REG);
...@@ -423,7 +423,7 @@ static ide_startstop_t idescsi_pc_intr (ide_drive_t *drive) ...@@ -423,7 +423,7 @@ static ide_startstop_t idescsi_pc_intr (ide_drive_t *drive)
} }
/* Clear the interrupt */ /* Clear the interrupt */
stat = drive->hwif->INB(IDE_STATUS_REG); stat = ide_read_status(drive);
if ((stat & DRQ_STAT) == 0) { if ((stat & DRQ_STAT) == 0) {
/* No more interrupts */ /* No more interrupts */
......
...@@ -18,14 +18,14 @@ ...@@ -18,14 +18,14 @@
#define MB_NO 7 /* media bay contains nothing */ #define MB_NO 7 /* media bay contains nothing */
int check_media_bay(struct device_node *which_bay, int what); int check_media_bay(struct device_node *which_bay, int what);
int check_media_bay_by_base(unsigned long base, int what);
/* Number of bays in the machine or 0 */ /* Number of bays in the machine or 0 */
extern int media_bay_count; extern int media_bay_count;
/* called by pmac-ide.c to register IDE controller for media bay */ int check_media_bay_by_base(unsigned long base, int what);
extern int media_bay_set_ide_infos(struct device_node* which_bay, /* called by IDE PMAC host driver to register IDE controller for media bay */
unsigned long base, int irq, int index); int media_bay_set_ide_infos(struct device_node *which_bay, unsigned long base,
int irq, int index);
#endif /* __KERNEL__ */ #endif /* __KERNEL__ */
#endif /* _PPC_MEDIABAY_H */ #endif /* _PPC_MEDIABAY_H */
...@@ -115,10 +115,6 @@ typedef unsigned char byte; /* used everywhere */ ...@@ -115,10 +115,6 @@ typedef unsigned char byte; /* used everywhere */
#define SATA_ERROR_OFFSET (1) #define SATA_ERROR_OFFSET (1)
#define SATA_CONTROL_OFFSET (2) #define SATA_CONTROL_OFFSET (2)
#define SATA_MISC_OFFSET (0)
#define SATA_PHY_OFFSET (1)
#define SATA_IEN_OFFSET (2)
/* /*
* Our Physical Region Descriptor (PRD) table should be large enough * Our Physical Region Descriptor (PRD) table should be large enough
* to handle the biggest I/O request we are likely to see. Since requests * to handle the biggest I/O request we are likely to see. Since requests
...@@ -173,7 +169,7 @@ enum { ide_unknown, ide_generic, ide_pci, ...@@ -173,7 +169,7 @@ enum { ide_unknown, ide_generic, ide_pci,
ide_rz1000, ide_trm290, ide_rz1000, ide_trm290,
ide_cmd646, ide_cy82c693, ide_4drives, ide_cmd646, ide_cy82c693, ide_4drives,
ide_pmac, ide_etrax100, ide_acorn, ide_pmac, ide_etrax100, ide_acorn,
ide_au1xxx, ide_forced ide_au1xxx, ide_palm3710, ide_forced
}; };
typedef u8 hwif_chipset_t; typedef u8 hwif_chipset_t;
...@@ -198,17 +194,6 @@ struct ide_drive_s; ...@@ -198,17 +194,6 @@ struct ide_drive_s;
int ide_register_hw(hw_regs_t *, void (*)(struct ide_drive_s *), int ide_register_hw(hw_regs_t *, void (*)(struct ide_drive_s *),
struct hwif_s **); struct hwif_s **);
void ide_setup_ports( hw_regs_t *hw,
unsigned long base,
int *offsets,
unsigned long ctrl,
unsigned long intr,
ide_ack_intr_t *ack_intr,
#if 0
ide_io_ops_t *iops,
#endif
int irq);
static inline void ide_std_init_ports(hw_regs_t *hw, static inline void ide_std_init_ports(hw_regs_t *hw,
unsigned long io_addr, unsigned long io_addr,
unsigned long ctl_addr) unsigned long ctl_addr)
...@@ -473,7 +458,6 @@ typedef struct hwif_s { ...@@ -473,7 +458,6 @@ typedef struct hwif_s {
/* task file registers for pata and sata */ /* task file registers for pata and sata */
unsigned long io_ports[IDE_NR_PORTS]; unsigned long io_ports[IDE_NR_PORTS];
unsigned long sata_scr[SATA_NR_PORTS]; unsigned long sata_scr[SATA_NR_PORTS];
unsigned long sata_misc[SATA_NR_PORTS];
ide_drive_t drives[MAX_DRIVES]; /* drive info */ ide_drive_t drives[MAX_DRIVES]; /* drive info */
...@@ -1014,7 +998,8 @@ extern int __ide_pci_register_driver(struct pci_driver *driver, struct module *o ...@@ -1014,7 +998,8 @@ extern int __ide_pci_register_driver(struct pci_driver *driver, struct module *o
void ide_pci_setup_ports(struct pci_dev *, const struct ide_port_info *, int, u8 *); void ide_pci_setup_ports(struct pci_dev *, const struct ide_port_info *, int, u8 *);
void ide_setup_pci_noise(struct pci_dev *, const struct ide_port_info *); void ide_setup_pci_noise(struct pci_dev *, const struct ide_port_info *);
#ifdef CONFIG_BLK_DEV_IDEDMA_PCI /* FIXME: palm_bk3710 uses BLK_DEV_IDEDMA_PCI without BLK_DEV_IDEPCI! */
#if defined(CONFIG_BLK_DEV_IDEPCI) && defined(CONFIG_BLK_DEV_IDEDMA_PCI)
void ide_hwif_setup_dma(ide_hwif_t *, const struct ide_port_info *); void ide_hwif_setup_dma(ide_hwif_t *, const struct ide_port_info *);
#else #else
static inline void ide_hwif_setup_dma(ide_hwif_t *hwif, static inline void ide_hwif_setup_dma(ide_hwif_t *hwif,
...@@ -1324,4 +1309,25 @@ static inline void ide_set_irq(ide_drive_t *drive, int on) ...@@ -1324,4 +1309,25 @@ static inline void ide_set_irq(ide_drive_t *drive, int on)
drive->hwif->OUTB(drive->ctl | (on ? 0 : 2), IDE_CONTROL_REG); drive->hwif->OUTB(drive->ctl | (on ? 0 : 2), IDE_CONTROL_REG);
} }
static inline u8 ide_read_status(ide_drive_t *drive)
{
ide_hwif_t *hwif = drive->hwif;
return hwif->INB(hwif->io_ports[IDE_STATUS_OFFSET]);
}
static inline u8 ide_read_altstatus(ide_drive_t *drive)
{
ide_hwif_t *hwif = drive->hwif;
return hwif->INB(hwif->io_ports[IDE_CONTROL_OFFSET]);
}
static inline u8 ide_read_error(ide_drive_t *drive)
{
ide_hwif_t *hwif = drive->hwif;
return hwif->INB(hwif->io_ports[IDE_ERROR_OFFSET]);
}
#endif /* _IDE_H */ #endif /* _IDE_H */
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册