⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 ide.c

📁 《嵌入式系统设计与实例开发实验教材II:基于ARM9微处理器与Linux操作系统》IDE—CF卡模块读写实验
💻 C
📖 第 1 页 / 共 5 页
字号:
/* *  linux/drivers/ide/ide.c		Version 6.31	June 9, 2000 * *  Copyright (C) 1994-1998  Linus Torvalds & authors (see below) *//* *  Mostly written by Mark Lord  <mlord@pobox.com> *                and Gadi Oxman <gadio@netvision.net.il> *                and Andre Hedrick <andre@linux-ide.org> * *  See linux/MAINTAINERS for address of current maintainer. * * This is the multiple IDE interface driver, as evolved from hd.c. * It supports up to MAX_HWIFS IDE interfaces, on one or more IRQs (usually 14 & 15). * There can be up to two drives per interface, as per the ATA-2 spec. * * Primary:    ide0, port 0x1f0; major=3;  hda is minor=0; hdb is minor=64 * Secondary:  ide1, port 0x170; major=22; hdc is minor=0; hdd is minor=64 * Tertiary:   ide2, port 0x???; major=33; hde is minor=0; hdf is minor=64 * Quaternary: ide3, port 0x???; major=34; hdg is minor=0; hdh is minor=64 * ... * *  From hd.c: *  | *  | It traverses the request-list, using interrupts to jump between functions. *  | As nearly all functions can be called within interrupts, we may not sleep. *  | Special care is recommended.  Have Fun! *  | *  | modified by Drew Eckhardt to check nr of hd's from the CMOS. *  | *  | Thanks to Branko Lankester, lankeste@fwi.uva.nl, who found a bug *  | in the early extended-partition checks and added DM partitions. *  | *  | Early work on error handling by Mika Liljeberg (liljeber@cs.Helsinki.FI). *  | *  | IRQ-unmask, drive-id, multiple-mode, support for ">16 heads", *  | and general streamlining by Mark Lord (mlord@pobox.com). * *  October, 1994 -- Complete line-by-line overhaul for linux 1.1.x, by: * *	Mark Lord	(mlord@pobox.com)		(IDE Perf.Pkg) *	Delman Lee	(delman@ieee.org)		("Mr. atdisk2") *	Scott Snyder	(snyder@fnald0.fnal.gov)	(ATAPI IDE cd-rom) * *  This was a rewrite of just about everything from hd.c, though some original *  code is still sprinkled about.  Think of it as a major evolution, with *  inspiration from lots of linux users, esp.  hamish@zot.apana.org.au * *  Version 1.0 ALPHA	initial code, primary i/f working okay *  Version 1.3 BETA	dual i/f on shared irq tested & working! *  Version 1.4 BETA	added auto probing for irq(s) *  Version 1.5 BETA	added ALPHA (untested) support for IDE cd-roms, *  ... * Version 5.50		allow values as small as 20 for idebus= * Version 5.51		force non io_32bit in drive_cmd_intr() *			change delay_10ms() to delay_50ms() to fix problems * Version 5.52		fix incorrect invalidation of removable devices *			add "hdx=slow" command line option * Version 5.60		start to modularize the driver; the disk and ATAPI *			 drivers can be compiled as loadable modules. *			move IDE probe code to ide-probe.c *			move IDE disk code to ide-disk.c *			add support for generic IDE device subdrivers *			add m68k code from Geert Uytterhoeven *			probe all interfaces by default *			add ioctl to (re)probe an interface * Version 6.00		use per device request queues *			attempt to optimize shared hwgroup performance *			add ioctl to manually adjust bandwidth algorithms *			add kerneld support for the probe module *			fix bug in ide_error() *			fix bug in the first ide_get_lock() call for Atari *			don't flush leftover data for ATAPI devices * Version 6.01		clear hwgroup->active while the hwgroup sleeps *			support HDIO_GETGEO for floppies * Version 6.02		fix ide_ack_intr() call *			check partition table on floppies * Version 6.03		handle bad status bit sequencing in ide_wait_stat() * Version 6.10		deleted old entries from this list of updates *			replaced triton.c with ide-dma.c generic PCI DMA *			added support for BIOS-enabled UltraDMA *			rename all "promise" things to "pdc4030" *			fix EZ-DRIVE handling on small disks * Version 6.11		fix probe error in ide_scan_devices() *			fix ancient "jiffies" polling bugs *			mask all hwgroup interrupts on each irq entry * Version 6.12		integrate ioctl and proc interfaces *			fix parsing of "idex=" command line parameter * Version 6.13		add support for ide4/ide5 courtesy rjones@orchestream.com * Version 6.14		fixed IRQ sharing among PCI devices * Version 6.15		added SMP awareness to IDE drivers * Version 6.16		fixed various bugs; even more SMP friendly * Version 6.17		fix for newest EZ-Drive problem * Version 6.18		default unpartitioned-disk translation now "BIOS LBA" * Version 6.19		Re-design for a UNIFORM driver for all platforms, *			  model based on suggestions from Russell King and *			  Geert Uytterhoeven *			Promise DC4030VL now supported. *			add support for ide6/ide7 *			delay_50ms() changed to ide_delay_50ms() and exported. * Version 6.20		Added/Fixed Generic ATA-66 support and hwif detection. *			Added hdx=flash to allow for second flash disk *			  detection w/o the hang loop. *			Added support for ide8/ide9 *			Added idex=ata66 for the quirky chipsets that are *			  ATA-66 compliant, but have yet to determine a method *			  of verification of the 80c cable presence. *			  Specifically Promise's PDC20262 chipset. * Version 6.21		Fixing/Fixed SMP spinlock issue with insight from an old *			  hat that clarified original low level driver design. * Version 6.30		Added SMP support; fixed multmode issues.  -ml * Version 6.31		Debug Share INTR's and request queue streaming *			Native ATA-100 support *			Prep for Cascades Project * *  Some additional driver compile-time options are in ./include/linux/ide.h * *  To do, in likely order of completion: *	- modify kernel to obtain BIOS geometry for drives on 2nd/3rd/4th i/f * */#define	REVISION	"Revision: 6.31"#define	VERSION		"Id: ide.c 6.31 2000/06/09"#undef REALLY_SLOW_IO		/* most systems can safely undef this */#define _IDE_C			/* Tell ide.h it's really us */#include <linux/config.h>#include <linux/module.h>#include <linux/types.h>#include <linux/string.h>#include <linux/kernel.h>#include <linux/timer.h>#include <linux/mm.h>#include <linux/interrupt.h>#include <linux/major.h>#include <linux/errno.h>#include <linux/genhd.h>#include <linux/blkpg.h>#include <linux/slab.h>#ifndef MODULE#include <linux/init.h>#endif /* MODULE */#include <linux/pci.h>#include <linux/delay.h>#include <linux/ide.h>#include <linux/devfs_fs_kernel.h>#include <linux/completion.h>#include <asm/byteorder.h>#include <asm/irq.h>#include <asm/uaccess.h>#include <asm/io.h>#include <asm/bitops.h>#include "ide_modes.h"#ifdef CONFIG_KMOD#include <linux/kmod.h>#endif /* CONFIG_KMOD *//* default maximum number of failures */#define IDE_DEFAULT_MAX_FAILURES 	1static const byte ide_hwif_to_major[] = { IDE0_MAJOR, IDE1_MAJOR, IDE2_MAJOR, IDE3_MAJOR, IDE4_MAJOR, IDE5_MAJOR, IDE6_MAJOR, IDE7_MAJOR, IDE8_MAJOR, IDE9_MAJOR };static int	idebus_parameter; /* holds the "idebus=" parameter */static int	system_bus_speed; /* holds what we think is VESA/PCI bus speed */static int	initializing;     /* set while initializing built-in drivers */#ifdef CONFIG_BLK_DEV_IDEPCIstatic int	ide_scan_direction;	/* THIS was formerly 2.2.x pci=reverse */#endif /* CONFIG_BLK_DEV_IDEPCI */#if defined(__mc68000__) || defined(CONFIG_APUS)/* * ide_lock is used by the Atari code to obtain access to the IDE interrupt, * which is shared between several drivers. */static int	ide_lock;#endif /* __mc68000__ || CONFIG_APUS */int noautodma = 0;/* * ide_modules keeps track of the available IDE chipset/probe/driver modules. */ide_module_t *ide_modules;ide_module_t *ide_probe;/* * This is declared extern in ide.h, for access by other IDE modules: */ide_hwif_t	ide_hwifs[MAX_HWIFS];	/* master data repository */#if (DISK_RECOVERY_TIME > 0)/* * For really screwy hardware (hey, at least it *can* be used with Linux) * we can enforce a minimum delay time between successive operations. */static unsigned long read_timer (void){	unsigned long t, flags;	int i;	__save_flags(flags);	/* local CPU only */	__cli();		/* local CPU only */	t = jiffies * 11932;    	outb_p(0, 0x43);	i = inb_p(0x40);	i |= inb(0x40) << 8;	__restore_flags(flags);	/* local CPU only */	return (t - i);}#endif /* DISK_RECOVERY_TIME */static inline void set_recovery_timer (ide_hwif_t *hwif){#if (DISK_RECOVERY_TIME > 0)	hwif->last_time = read_timer();#endif /* DISK_RECOVERY_TIME */}/* * Do not even *think* about calling this! */static void init_hwif_data (unsigned int index){	unsigned int unit;	ide_hwif_t *hwif = &ide_hwifs[index];	/* bulk initialize hwif & drive info with zeros */	memset(hwif, 0, sizeof(ide_hwif_t));	/* fill in any non-zero initial values */	hwif->index     = index;	hwif->noprobe	= 1;	hwif->major	= ide_hwif_to_major[index];	hwif->name[0]	= 'i';	hwif->name[1]	= 'd';	hwif->name[2]	= 'e';	hwif->name[3]	= '0' + index;	hwif->bus_state = BUSSTATE_ON;	for (unit = 0; unit < MAX_DRIVES; ++unit) {		ide_drive_t *drive = &hwif->drives[unit];		drive->media			= ide_disk;		drive->select.all		= (unit<<4)|0xa0;		drive->hwif			= hwif;		drive->ctl			= 0x08;		drive->ready_stat		= READY_STAT;		drive->bad_wstat		= BAD_W_STAT;		drive->special.b.recalibrate	= 1;		drive->special.b.set_geometry	= 1;		drive->name[0]			= 'h';		drive->name[1]			= 'd';		drive->name[2]			= 'a' + (index * MAX_DRIVES) + unit;		drive->max_failures		= IDE_DEFAULT_MAX_FAILURES;		init_waitqueue_head(&drive->wqueue);	}}/* * Old compatability function - initialise ports using ide_default_io_base */static void ide_old_init_default_hwifs(void){	unsigned int index;	ide_ioreg_t base;	ide_hwif_t *hwif;	for (index = 0; index < MAX_HWIFS; index++) {		hwif = &ide_hwifs[index];		base = ide_default_io_base(index);		if (base) {			ide_init_hwif_ports(&hwif->hw, base, 0, &hwif->hw.irq);			memcpy(hwif->io_ports, hwif->hw.io_ports, sizeof(hwif->hw.io_ports));			hwif->noprobe = 0;		}	}}/* * init_ide_data() sets reasonable default values into all fields * of all instances of the hwifs and drives, but only on the first call. * Subsequent calls have no effect (they don't wipe out anything). * * This routine is normally called at driver initialization time, * but may also be called MUCH earlier during kernel "command-line" * parameter processing.  As such, we cannot depend on any other parts * of the kernel (such as memory allocation) to be functioning yet. * * This is too bad, as otherwise we could dynamically allocate the * ide_drive_t structs as needed, rather than always consuming memory * for the max possible number (MAX_HWIFS * MAX_DRIVES) of them. */#define MAGIC_COOKIE 0x12345678static void __init init_ide_data (void){	unsigned int index;	static unsigned long magic_cookie = MAGIC_COOKIE;	if (magic_cookie != MAGIC_COOKIE)		return;		/* already initialized */	magic_cookie = 0;	/* Initialise all interface structures */	for (index = 0; index < MAX_HWIFS; ++index)		init_hwif_data(index);	/* Add default hw interfaces */	ide_old_init_default_hwifs();	ide_init_default_hwifs();#ifdef CONFIG_BLK_DEV_HD	/* Check for any clashes with hd.c driver */	for (index = 0; index < MAX_HWIFS; ++index)		if (ide_hwifs[index].hw.io_ports[IDE_DATA_OFFSET] == HD_DATA)			hwif->noprobe = 1; /* may be overridden by ide_setup() */#endif /* CONFIG_BLK_DEV_HD */	idebus_parameter = 0;	system_bus_speed = 0;}/* * CompactFlash cards and their brethern pretend to be removable hard disks, except: *	(1) they never have a slave unit, and *	(2) they don't have doorlock mechanisms. * This test catches them, and is invoked elsewhere when setting appropriate config bits. * * FIXME: This treatment is probably applicable for *all* PCMCIA (PC CARD) devices, * so in linux 2.3.x we should change this to just treat all PCMCIA drives this way, * and get rid of the model-name tests below (too big of an interface change for 2.2.x). * At that time, we might also consider parameterizing the timeouts and retries, * since these are MUCH faster than mechanical drives.	-M.Lord */int drive_is_flashcard (ide_drive_t *drive){	struct hd_driveid *id = drive->id;	if (drive->removable && id != NULL) {		if (id->config == 0x848a) return 1;	/* CompactFlash */		if (!strncmp(id->model, "KODAK ATA_FLASH", 15)	/* Kodak */		 || !strncmp(id->model, "Hitachi CV", 10)	/* Hitachi */		 || !strncmp(id->model, "SunDisk SDCFB", 13)	/* SunDisk */		 || !strncmp(id->model, "HAGIWARA HPC", 12)	/* Hagiwara */		 || !strncmp(id->model, "LEXAR ATA_FLASH", 15)	/* Lexar */		 || !strncmp(id->model, "ATA_FLASH", 9))	/* Simple Tech */		{			return 1;	/* yes, it is a flash memory card */		}	}	return 0;	/* no, it is not a flash memory card */}/* * ide_system_bus_speed() returns what we think is the system VESA/PCI * bus speed (in MHz).  This is used for calculating interface PIO timings. * The default is 40 for known PCI systems, 50 otherwise. * The "idebus=xx" parameter can be used to override this value. * The actual value to be used is computed/displayed the first time through. */int ide_system_bus_speed (void){	if (!system_bus_speed) {		if (idebus_parameter)			system_bus_speed = idebus_parameter;	/* user supplied value */#ifdef CONFIG_PCI		else if (pci_present())			system_bus_speed = 33;	/* safe default value for PCI */#endif /* CONFIG_PCI */		else			system_bus_speed = 50;	/* safe default value for VESA and PCI */		printk("ide: Assuming %dMHz system bus speed for PIO modes%s\n", system_bus_speed,			idebus_parameter ? "" : "; override with idebus=xx");	}	return system_bus_speed;}#if SUPPORT_VLB_SYNC/* * Some localbus EIDE interfaces require a special access sequence * when using 32-bit I/O instructions to transfer data.  We call this * the "vlb_sync" sequence, which consists of three successive reads * of the sector count register location, with interrupts disabled * to ensure that the reads all happen together. */static inline void do_vlb_sync (ide_ioreg_t port) {	(void) inb (port);	(void) inb (port);	(void) inb (port);}#endif /* SUPPORT_VLB_SYNC *//* * This is used for most PIO data transfers *from* the IDE interface */void ide_input_data (ide_drive_t *drive, void *buffer, unsigned int wcount){	byte io_32bit;	/* first check if this controller has defined a special function	 * for handling polled ide transfers	 */	if(HWIF(drive)->ideproc) {		HWIF(drive)->ideproc(ideproc_ide_input_data,				     drive, buffer, wcount);		return;	}	io_32bit = drive->io_32bit;

⌨️ 快捷键说明

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