ide-lib.c

来自「linux 内核源代码」· C语言 代码 · 共 622 行 · 第 1/2 页

C
622
字号
}EXPORT_SYMBOL_GPL(ide_get_best_pio_mode);/* req_pio == "255" for auto-tune */void ide_set_pio(ide_drive_t *drive, u8 req_pio){	ide_hwif_t *hwif = drive->hwif;	u8 host_pio, pio;	if (hwif->set_pio_mode == NULL)		return;	BUG_ON(hwif->pio_mask == 0x00);	host_pio = fls(hwif->pio_mask) - 1;	pio = ide_get_best_pio_mode(drive, req_pio, host_pio);	/*	 * TODO:	 * - report device max PIO mode	 * - check req_pio != 255 against device max PIO mode	 */	printk(KERN_DEBUG "%s: host max PIO%d wanted PIO%d%s selected PIO%d\n",			  drive->name, host_pio, req_pio,			  req_pio == 255 ? "(auto-tune)" : "", pio);	(void)ide_set_pio_mode(drive, XFER_PIO_0 + pio);}EXPORT_SYMBOL_GPL(ide_set_pio);/** *	ide_toggle_bounce	-	handle bounce buffering *	@drive: drive to update *	@on: on/off boolean * *	Enable or disable bounce buffering for the device. Drives move *	between PIO and DMA and that changes the rules we need. */ void ide_toggle_bounce(ide_drive_t *drive, int on){	u64 addr = BLK_BOUNCE_HIGH;	/* dma64_addr_t */	if (!PCI_DMA_BUS_IS_PHYS) {		addr = BLK_BOUNCE_ANY;	} else if (on && drive->media == ide_disk) {		if (HWIF(drive)->pci_dev)			addr = HWIF(drive)->pci_dev->dma_mask;	}	if (drive->queue)		blk_queue_bounce_limit(drive->queue, addr);}int ide_set_pio_mode(ide_drive_t *drive, const u8 mode){	ide_hwif_t *hwif = drive->hwif;	if (hwif->set_pio_mode == NULL)		return -1;	/*	 * TODO: temporary hack for some legacy host drivers that didn't	 * set transfer mode on the device in ->set_pio_mode method...	 */	if (hwif->set_dma_mode == NULL) {		hwif->set_pio_mode(drive, mode - XFER_PIO_0);		return 0;	}	if (hwif->host_flags & IDE_HFLAG_POST_SET_MODE) {		if (ide_config_drive_speed(drive, mode))			return -1;		hwif->set_pio_mode(drive, mode - XFER_PIO_0);		return 0;	} else {		hwif->set_pio_mode(drive, mode - XFER_PIO_0);		return ide_config_drive_speed(drive, mode);	}}int ide_set_dma_mode(ide_drive_t *drive, const u8 mode){	ide_hwif_t *hwif = drive->hwif;	if (hwif->set_dma_mode == NULL)		return -1;	if (hwif->host_flags & IDE_HFLAG_POST_SET_MODE) {		if (ide_config_drive_speed(drive, mode))			return -1;		hwif->set_dma_mode(drive, mode);		return 0;	} else {		hwif->set_dma_mode(drive, mode);		return ide_config_drive_speed(drive, mode);	}}EXPORT_SYMBOL_GPL(ide_set_dma_mode);/** *	ide_set_xfer_rate	-	set transfer rate *	@drive: drive to set *	@rate: speed to attempt to set *	 *	General helper for setting the speed of an IDE device. This *	function knows about user enforced limits from the configuration *	which ->set_pio_mode/->set_dma_mode does not. */int ide_set_xfer_rate(ide_drive_t *drive, u8 rate){	ide_hwif_t *hwif = drive->hwif;	if (hwif->set_dma_mode == NULL)		return -1;	rate = ide_rate_filter(drive, rate);	if (rate >= XFER_PIO_0 && rate <= XFER_PIO_5)		return ide_set_pio_mode(drive, rate);	/*	 * TODO: transfer modes 0x00-0x07 passed from the user-space are	 * currently handled here which needs fixing (please note that such	 * case could happen iff the transfer mode has already been set on	 * the device by ide-proc.c::set_xfer_rate()).	 */	return ide_set_dma_mode(drive, rate);}static void ide_dump_opcode(ide_drive_t *drive){	struct request *rq;	u8 opcode = 0;	int found = 0;	spin_lock(&ide_lock);	rq = NULL;	if (HWGROUP(drive))		rq = HWGROUP(drive)->rq;	spin_unlock(&ide_lock);	if (!rq)		return;	if (rq->cmd_type == REQ_TYPE_ATA_CMD ||	    rq->cmd_type == REQ_TYPE_ATA_TASK) {		char *args = rq->buffer;		if (args) {			opcode = args[0];			found = 1;		}	} else if (rq->cmd_type == REQ_TYPE_ATA_TASKFILE) {		ide_task_t *args = rq->special;		if (args) {			task_struct_t *tf = (task_struct_t *) args->tfRegister;			opcode = tf->command;			found = 1;		}	}	printk("ide: failed opcode was: ");	if (!found)		printk("unknown\n");	else		printk("0x%02x\n", opcode);}static u8 ide_dump_ata_status(ide_drive_t *drive, const char *msg, u8 stat){	ide_hwif_t *hwif = HWIF(drive);	unsigned long flags;	u8 err = 0;	local_irq_save(flags);	printk("%s: %s: status=0x%02x { ", drive->name, msg, stat);	if (stat & BUSY_STAT)		printk("Busy ");	else {		if (stat & READY_STAT)	printk("DriveReady ");		if (stat & WRERR_STAT)	printk("DeviceFault ");		if (stat & SEEK_STAT)	printk("SeekComplete ");		if (stat & DRQ_STAT)	printk("DataRequest ");		if (stat & ECC_STAT)	printk("CorrectedError ");		if (stat & INDEX_STAT)	printk("Index ");		if (stat & ERR_STAT)	printk("Error ");	}	printk("}\n");	if ((stat & (BUSY_STAT|ERR_STAT)) == ERR_STAT) {		err = hwif->INB(IDE_ERROR_REG);		printk("%s: %s: error=0x%02x { ", drive->name, msg, err);		if (err & ABRT_ERR)	printk("DriveStatusError ");		if (err & ICRC_ERR)			printk((err & ABRT_ERR) ? "BadCRC " : "BadSector ");		if (err & ECC_ERR)	printk("UncorrectableError ");		if (err & ID_ERR)	printk("SectorIdNotFound ");		if (err & TRK0_ERR)	printk("TrackZeroNotFound ");		if (err & MARK_ERR)	printk("AddrMarkNotFound ");		printk("}");		if ((err & (BBD_ERR | ABRT_ERR)) == BBD_ERR ||		    (err & (ECC_ERR|ID_ERR|MARK_ERR))) {			if (drive->addressing == 1) {				__u64 sectors = 0;				u32 low = 0, high = 0;				hwif->OUTB(drive->ctl&~0x80, IDE_CONTROL_REG);				low = ide_read_24(drive);				hwif->OUTB(drive->ctl|0x80, IDE_CONTROL_REG);				high = ide_read_24(drive);				sectors = ((__u64)high << 24) | low;				printk(", LBAsect=%llu, high=%d, low=%d",				       (unsigned long long) sectors,				       high, low);			} else {				u8 cur = hwif->INB(IDE_SELECT_REG);				if (cur & 0x40) {	/* using LBA? */					printk(", LBAsect=%ld", (unsigned long)					 ((cur&0xf)<<24)					 |(hwif->INB(IDE_HCYL_REG)<<16)					 |(hwif->INB(IDE_LCYL_REG)<<8)					 | hwif->INB(IDE_SECTOR_REG));				} else {					printk(", CHS=%d/%d/%d",					 (hwif->INB(IDE_HCYL_REG)<<8) +					  hwif->INB(IDE_LCYL_REG),					  cur & 0xf,					  hwif->INB(IDE_SECTOR_REG));				}			}			if (HWGROUP(drive) && HWGROUP(drive)->rq)				printk(", sector=%llu",					(unsigned long long)HWGROUP(drive)->rq->sector);		}		printk("\n");	}	ide_dump_opcode(drive);	local_irq_restore(flags);	return err;}/** *	ide_dump_atapi_status       -       print human readable atapi status *	@drive: drive that status applies to *	@msg: text message to print *	@stat: status byte to decode * *	Error reporting, in human readable form (luxurious, but a memory hog). */static u8 ide_dump_atapi_status(ide_drive_t *drive, const char *msg, u8 stat){	unsigned long flags;	atapi_status_t status;	atapi_error_t error;	status.all = stat;	error.all = 0;	local_irq_save(flags);	printk("%s: %s: status=0x%02x { ", drive->name, msg, stat);	if (status.b.bsy)		printk("Busy ");	else {		if (status.b.drdy)	printk("DriveReady ");		if (status.b.df)	printk("DeviceFault ");		if (status.b.dsc)	printk("SeekComplete ");		if (status.b.drq)	printk("DataRequest ");		if (status.b.corr)	printk("CorrectedError ");		if (status.b.idx)	printk("Index ");		if (status.b.check)	printk("Error ");	}	printk("}\n");	if (status.b.check && !status.b.bsy) {		error.all = HWIF(drive)->INB(IDE_ERROR_REG);		printk("%s: %s: error=0x%02x { ", drive->name, msg, error.all);		if (error.b.ili)	printk("IllegalLengthIndication ");		if (error.b.eom)	printk("EndOfMedia ");		if (error.b.abrt)	printk("AbortedCommand ");		if (error.b.mcr)	printk("MediaChangeRequested ");		if (error.b.sense_key)	printk("LastFailedSense=0x%02x ",						error.b.sense_key);		printk("}\n");	}	ide_dump_opcode(drive);	local_irq_restore(flags);	return error.all;}/** *	ide_dump_status		-	translate ATA/ATAPI error *	@drive: drive the error occured on *	@msg: information string *	@stat: status byte * *	Error reporting, in human readable form (luxurious, but a memory hog). *	Combines the drive name, message and status byte to provide a *	user understandable explanation of the device error. */u8 ide_dump_status(ide_drive_t *drive, const char *msg, u8 stat){	if (drive->media == ide_disk)		return ide_dump_ata_status(drive, msg, stat);	return ide_dump_atapi_status(drive, msg, stat);}EXPORT_SYMBOL(ide_dump_status);

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?