2005-04-17 06:20:36 +08:00
|
|
|
/*
|
2005-08-29 08:18:39 +08:00
|
|
|
* libata.h - helper library for ATA
|
|
|
|
*
|
|
|
|
* Copyright 2003-2004 Red Hat, Inc. All rights reserved.
|
|
|
|
* Copyright 2003-2004 Jeff Garzik
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* 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, 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; see the file COPYING. If not, write to
|
|
|
|
* the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* libata documentation is available via 'make {ps|pdf}docs',
|
|
|
|
* as Documentation/DocBook/libata.*
|
|
|
|
*
|
2005-04-17 06:20:36 +08:00
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef __LIBATA_H__
|
|
|
|
#define __LIBATA_H__
|
|
|
|
|
|
|
|
#define DRV_NAME "libata"
|
2007-09-23 12:19:54 +08:00
|
|
|
#define DRV_VERSION "3.00" /* must be exactly four chars */
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
struct ata_scsi_args {
|
2006-01-06 16:56:18 +08:00
|
|
|
struct ata_device *dev;
|
2005-04-17 06:20:36 +08:00
|
|
|
u16 *id;
|
|
|
|
struct scsi_cmnd *cmd;
|
|
|
|
void (*done)(struct scsi_cmnd *);
|
|
|
|
};
|
|
|
|
|
|
|
|
/* libata-core.c */
|
2006-11-10 17:08:10 +08:00
|
|
|
enum {
|
|
|
|
/* flags for ata_dev_read_id() */
|
|
|
|
ATA_READID_POSTRESET = (1 << 0), /* reading ID after reset */
|
2007-02-02 15:22:30 +08:00
|
|
|
|
|
|
|
/* selector for ata_down_xfermask_limit() */
|
|
|
|
ATA_DNXFER_PIO = 0, /* speed down PIO */
|
|
|
|
ATA_DNXFER_DMA = 1, /* speed down DMA */
|
|
|
|
ATA_DNXFER_40C = 2, /* apply 40c cable limit */
|
|
|
|
ATA_DNXFER_FORCE_PIO = 3, /* force PIO */
|
|
|
|
ATA_DNXFER_FORCE_PIO0 = 4, /* force PIO0 */
|
|
|
|
|
|
|
|
ATA_DNXFER_QUIET = (1 << 31),
|
2006-11-10 17:08:10 +08:00
|
|
|
};
|
|
|
|
|
2007-04-17 22:44:07 +08:00
|
|
|
extern unsigned int ata_print_id;
|
2006-05-31 17:27:42 +08:00
|
|
|
extern struct workqueue_struct *ata_aux_wq;
|
2005-08-30 15:37:42 +08:00
|
|
|
extern int atapi_enabled;
|
2006-04-04 10:57:18 +08:00
|
|
|
extern int atapi_dmadir;
|
2007-08-08 00:08:45 +08:00
|
|
|
extern int atapi_passthru16;
|
2006-02-28 11:31:19 +08:00
|
|
|
extern int libata_fua;
|
2007-03-28 13:57:37 +08:00
|
|
|
extern int libata_noacpi;
|
2006-05-15 19:57:53 +08:00
|
|
|
extern struct ata_queued_cmd *ata_qc_new_init(struct ata_device *dev);
|
2006-11-14 21:47:10 +08:00
|
|
|
extern int ata_build_rw_tf(struct ata_taskfile *tf, struct ata_device *dev,
|
|
|
|
u64 block, u32 n_block, unsigned int tf_flags,
|
|
|
|
unsigned int tag);
|
2006-11-14 21:37:35 +08:00
|
|
|
extern u64 ata_tf_read_block(struct ata_taskfile *tf, struct ata_device *dev);
|
2006-05-15 19:57:53 +08:00
|
|
|
extern void ata_dev_disable(struct ata_device *dev);
|
2006-03-05 14:29:09 +08:00
|
|
|
extern void ata_port_flush_task(struct ata_port *ap);
|
2006-05-15 19:57:53 +08:00
|
|
|
extern unsigned ata_exec_internal(struct ata_device *dev,
|
2006-04-02 17:51:53 +08:00
|
|
|
struct ata_taskfile *tf, const u8 *cdb,
|
|
|
|
int dma_dir, void *buf, unsigned int buflen);
|
2006-11-14 21:47:09 +08:00
|
|
|
extern unsigned ata_exec_internal_sg(struct ata_device *dev,
|
|
|
|
struct ata_taskfile *tf, const u8 *cdb,
|
|
|
|
int dma_dir, struct scatterlist *sg,
|
|
|
|
unsigned int n_elem);
|
2006-06-24 19:30:19 +08:00
|
|
|
extern unsigned int ata_do_simple_cmd(struct ata_device *dev, u8 cmd);
|
2006-05-31 17:27:44 +08:00
|
|
|
extern int ata_dev_read_id(struct ata_device *dev, unsigned int *p_class,
|
2006-11-10 17:08:10 +08:00
|
|
|
unsigned int flags, u16 *id);
|
2007-05-15 02:28:15 +08:00
|
|
|
extern int ata_dev_reread_id(struct ata_device *dev, unsigned int readid_flags);
|
2007-09-23 12:14:12 +08:00
|
|
|
extern int ata_dev_revalidate(struct ata_device *dev, unsigned int new_class,
|
|
|
|
unsigned int readid_flags);
|
2006-11-01 17:38:52 +08:00
|
|
|
extern int ata_dev_configure(struct ata_device *dev);
|
2007-08-06 17:36:23 +08:00
|
|
|
extern int sata_down_spd_limit(struct ata_link *link);
|
|
|
|
extern int sata_set_spd_needed(struct ata_link *link);
|
2007-02-02 15:22:30 +08:00
|
|
|
extern int ata_down_xfermask_limit(struct ata_device *dev, unsigned int sel);
|
2007-08-06 17:36:23 +08:00
|
|
|
extern int ata_set_mode(struct ata_link *link, struct ata_device **r_failed_dev);
|
2006-11-14 21:47:10 +08:00
|
|
|
extern void ata_sg_clean(struct ata_queued_cmd *qc);
|
2005-04-17 06:20:36 +08:00
|
|
|
extern void ata_qc_free(struct ata_queued_cmd *qc);
|
2006-03-31 19:41:11 +08:00
|
|
|
extern void ata_qc_issue(struct ata_queued_cmd *qc);
|
2006-05-15 19:58:05 +08:00
|
|
|
extern void __ata_qc_complete(struct ata_queued_cmd *qc);
|
2005-04-17 06:20:36 +08:00
|
|
|
extern int ata_check_atapi_dma(struct ata_queued_cmd *qc);
|
|
|
|
extern void ata_dev_select(struct ata_port *ap, unsigned int device,
|
|
|
|
unsigned int wait, unsigned int can_sleep);
|
|
|
|
extern void swap_buf_le16(u16 *buf, unsigned int buf_words);
|
2006-06-24 19:30:19 +08:00
|
|
|
extern int ata_flush_cache(struct ata_device *dev);
|
2006-05-31 17:27:30 +08:00
|
|
|
extern void ata_dev_init(struct ata_device *dev);
|
2007-09-23 12:14:12 +08:00
|
|
|
extern void ata_link_init(struct ata_port *ap, struct ata_link *link, int pmp);
|
|
|
|
extern int sata_link_init_spd(struct ata_link *link);
|
2005-05-13 03:45:22 +08:00
|
|
|
extern int ata_task_ioctl(struct scsi_device *scsidev, void __user *arg);
|
|
|
|
extern int ata_cmd_ioctl(struct scsi_device *scsidev, void __user *arg);
|
2007-04-17 22:44:07 +08:00
|
|
|
extern struct ata_port *ata_port_alloc(struct ata_host *host);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2006-09-29 02:29:01 +08:00
|
|
|
/* libata-acpi.c */
|
2007-05-05 22:50:38 +08:00
|
|
|
#ifdef CONFIG_ATA_ACPI
|
2007-09-23 12:19:54 +08:00
|
|
|
extern void ata_acpi_associate_sata_port(struct ata_port *ap);
|
2007-05-15 02:28:16 +08:00
|
|
|
extern void ata_acpi_associate(struct ata_host *host);
|
2007-05-15 02:28:16 +08:00
|
|
|
extern int ata_acpi_on_suspend(struct ata_port *ap);
|
libata: reimplement ACPI invocation
This patch reimplements ACPI invocation such that, instead of
exporting ACPI details to the rest of libata, ACPI event handlers -
ata_acpi_on_resume() and ata_acpi_on_devcfg() - are used. These two
functions are responsible for determining whether specific ACPI method
is used and when.
On resume, _GTF is scheduled by setting ATA_DFLAG_ACPI_PENDING device
flag. This is done this way to avoid performing the action on wrong
device device (device swapping while suspended).
On every ata_dev_configure(), ata_acpi_on_devcfg() is called, which
performs _SDD and _GTF. _GTF is performed only after resuming and, if
SATA, hardreset as the ACPI spec specifies. As _GTF may contain
arbitrary commands, IDENTIFY page is re-read after _GTF taskfiles are
executed.
If one of ACPI methods fails, ata_acpi_on_devcfg() retries on the
first failure. If it fails again on the second try, ACPI is disabled
on the device. Note that successful configuration clears ACPI failed
status.
With all feature checks moved to the above two functions,
do_drive_set_taskfiles() is trivial and thus collapsed into
ata_acpi_exec_tfs(), which is now static and converted to return the
number of executed taskfiles to be used by ata_acpi_on_resume(). As
failures are handled properly, ata_acpi_push_id() now returns -errno
on errors instead of unconditional zero.
Signed-off-by: Tejun Heo <htejun@gmail.com>
Signed-off-by: Jeff Garzik <jeff@garzik.org>
2007-05-15 02:28:16 +08:00
|
|
|
extern void ata_acpi_on_resume(struct ata_port *ap);
|
|
|
|
extern int ata_acpi_on_devcfg(struct ata_device *adev);
|
2006-09-29 02:29:01 +08:00
|
|
|
#else
|
2007-09-23 12:19:54 +08:00
|
|
|
static inline void ata_acpi_associate_sata_port(struct ata_port *ap) { }
|
2007-05-15 02:28:16 +08:00
|
|
|
static inline void ata_acpi_associate(struct ata_host *host) { }
|
2007-05-15 02:28:16 +08:00
|
|
|
static inline int ata_acpi_on_suspend(struct ata_port *ap) { return 0; }
|
libata: reimplement ACPI invocation
This patch reimplements ACPI invocation such that, instead of
exporting ACPI details to the rest of libata, ACPI event handlers -
ata_acpi_on_resume() and ata_acpi_on_devcfg() - are used. These two
functions are responsible for determining whether specific ACPI method
is used and when.
On resume, _GTF is scheduled by setting ATA_DFLAG_ACPI_PENDING device
flag. This is done this way to avoid performing the action on wrong
device device (device swapping while suspended).
On every ata_dev_configure(), ata_acpi_on_devcfg() is called, which
performs _SDD and _GTF. _GTF is performed only after resuming and, if
SATA, hardreset as the ACPI spec specifies. As _GTF may contain
arbitrary commands, IDENTIFY page is re-read after _GTF taskfiles are
executed.
If one of ACPI methods fails, ata_acpi_on_devcfg() retries on the
first failure. If it fails again on the second try, ACPI is disabled
on the device. Note that successful configuration clears ACPI failed
status.
With all feature checks moved to the above two functions,
do_drive_set_taskfiles() is trivial and thus collapsed into
ata_acpi_exec_tfs(), which is now static and converted to return the
number of executed taskfiles to be used by ata_acpi_on_resume(). As
failures are handled properly, ata_acpi_push_id() now returns -errno
on errors instead of unconditional zero.
Signed-off-by: Tejun Heo <htejun@gmail.com>
Signed-off-by: Jeff Garzik <jeff@garzik.org>
2007-05-15 02:28:16 +08:00
|
|
|
static inline void ata_acpi_on_resume(struct ata_port *ap) { }
|
|
|
|
static inline int ata_acpi_on_devcfg(struct ata_device *adev) { return 0; }
|
2006-09-29 02:29:01 +08:00
|
|
|
#endif
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* libata-scsi.c */
|
2007-04-17 22:44:07 +08:00
|
|
|
extern int ata_scsi_add_hosts(struct ata_host *host,
|
|
|
|
struct scsi_host_template *sht);
|
2007-07-16 13:29:40 +08:00
|
|
|
extern void ata_scsi_scan_host(struct ata_port *ap, int sync);
|
2006-05-31 17:28:01 +08:00
|
|
|
extern int ata_scsi_offline_dev(struct ata_device *dev);
|
2007-09-23 12:14:13 +08:00
|
|
|
extern void ata_scsi_media_change_notify(struct ata_device *dev);
|
2006-11-22 22:55:48 +08:00
|
|
|
extern void ata_scsi_hotplug(struct work_struct *work);
|
2005-04-17 06:20:36 +08:00
|
|
|
extern unsigned int ata_scsiop_inq_std(struct ata_scsi_args *args, u8 *rbuf,
|
|
|
|
unsigned int buflen);
|
|
|
|
|
|
|
|
extern unsigned int ata_scsiop_inq_00(struct ata_scsi_args *args, u8 *rbuf,
|
|
|
|
unsigned int buflen);
|
|
|
|
|
|
|
|
extern unsigned int ata_scsiop_inq_80(struct ata_scsi_args *args, u8 *rbuf,
|
|
|
|
unsigned int buflen);
|
|
|
|
extern unsigned int ata_scsiop_inq_83(struct ata_scsi_args *args, u8 *rbuf,
|
|
|
|
unsigned int buflen);
|
|
|
|
extern unsigned int ata_scsiop_noop(struct ata_scsi_args *args, u8 *rbuf,
|
|
|
|
unsigned int buflen);
|
|
|
|
extern unsigned int ata_scsiop_sync_cache(struct ata_scsi_args *args, u8 *rbuf,
|
|
|
|
unsigned int buflen);
|
|
|
|
extern unsigned int ata_scsiop_mode_sense(struct ata_scsi_args *args, u8 *rbuf,
|
|
|
|
unsigned int buflen);
|
|
|
|
extern unsigned int ata_scsiop_read_cap(struct ata_scsi_args *args, u8 *rbuf,
|
|
|
|
unsigned int buflen);
|
|
|
|
extern unsigned int ata_scsiop_report_luns(struct ata_scsi_args *args, u8 *rbuf,
|
|
|
|
unsigned int buflen);
|
|
|
|
extern void ata_scsi_badcmd(struct scsi_cmnd *cmd,
|
|
|
|
void (*done)(struct scsi_cmnd *),
|
|
|
|
u8 asc, u8 ascq);
|
2005-10-09 20:55:41 +08:00
|
|
|
extern void ata_scsi_set_sense(struct scsi_cmnd *cmd,
|
|
|
|
u8 sk, u8 asc, u8 ascq);
|
2005-08-01 01:13:24 +08:00
|
|
|
extern void ata_scsi_rbuf_fill(struct ata_scsi_args *args,
|
2005-04-17 06:20:36 +08:00
|
|
|
unsigned int (*actor) (struct ata_scsi_args *args,
|
|
|
|
u8 *rbuf, unsigned int buflen));
|
2006-05-15 19:58:07 +08:00
|
|
|
extern void ata_schedule_scsi_eh(struct Scsi_Host *shost);
|
2006-11-22 22:55:48 +08:00
|
|
|
extern void ata_scsi_dev_rescan(struct work_struct *work);
|
2006-08-08 03:27:31 +08:00
|
|
|
extern int ata_bus_probe(struct ata_port *ap);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2007-09-23 12:19:54 +08:00
|
|
|
/* libata-pmp.c */
|
|
|
|
extern int sata_pmp_scr_read(struct ata_link *link, int reg, u32 *val);
|
|
|
|
extern int sata_pmp_scr_write(struct ata_link *link, int reg, u32 val);
|
|
|
|
extern int sata_pmp_attach(struct ata_device *dev);
|
|
|
|
|
2006-04-02 17:51:53 +08:00
|
|
|
/* libata-eh.c */
|
|
|
|
extern enum scsi_eh_timer_return ata_scsi_timed_out(struct scsi_cmnd *cmd);
|
2006-04-12 01:04:39 +08:00
|
|
|
extern void ata_scsi_error(struct Scsi_Host *host);
|
2006-05-31 17:27:27 +08:00
|
|
|
extern void ata_port_wait_eh(struct ata_port *ap);
|
2007-07-16 13:29:41 +08:00
|
|
|
extern void ata_eh_fastdrain_timerfn(unsigned long arg);
|
2006-05-15 19:58:05 +08:00
|
|
|
extern void ata_qc_schedule_eh(struct ata_queued_cmd *qc);
|
2007-09-23 12:14:12 +08:00
|
|
|
extern void ata_eh_detach_dev(struct ata_device *dev);
|
|
|
|
extern void ata_eh_about_to_do(struct ata_link *link, struct ata_device *dev,
|
|
|
|
unsigned int action);
|
|
|
|
extern void ata_eh_done(struct ata_link *link, struct ata_device *dev,
|
|
|
|
unsigned int action);
|
|
|
|
extern void ata_eh_autopsy(struct ata_port *ap);
|
|
|
|
extern void ata_eh_report(struct ata_port *ap);
|
|
|
|
extern int ata_eh_reset(struct ata_link *link, int classify,
|
|
|
|
ata_prereset_fn_t prereset, ata_reset_fn_t softreset,
|
|
|
|
ata_reset_fn_t hardreset, ata_postreset_fn_t postreset);
|
|
|
|
extern int ata_eh_recover(struct ata_port *ap, ata_prereset_fn_t prereset,
|
|
|
|
ata_reset_fn_t softreset, ata_reset_fn_t hardreset,
|
|
|
|
ata_postreset_fn_t postreset,
|
|
|
|
struct ata_link **r_failed_disk);
|
|
|
|
extern void ata_eh_finish(struct ata_port *ap);
|
2006-04-02 17:51:53 +08:00
|
|
|
|
2006-10-09 10:10:26 +08:00
|
|
|
/* libata-sff.c */
|
|
|
|
extern u8 ata_irq_on(struct ata_port *ap);
|
|
|
|
|
2007-01-30 16:59:17 +08:00
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
#endif /* __LIBATA_H__ */
|