1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0-only
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun * ATAPI support.
4*4882a593Smuzhiyun */
5*4882a593Smuzhiyun
6*4882a593Smuzhiyun #include <linux/kernel.h>
7*4882a593Smuzhiyun #include <linux/cdrom.h>
8*4882a593Smuzhiyun #include <linux/delay.h>
9*4882a593Smuzhiyun #include <linux/export.h>
10*4882a593Smuzhiyun #include <linux/ide.h>
11*4882a593Smuzhiyun #include <linux/scatterlist.h>
12*4882a593Smuzhiyun #include <linux/gfp.h>
13*4882a593Smuzhiyun
14*4882a593Smuzhiyun #include <scsi/scsi.h>
15*4882a593Smuzhiyun
16*4882a593Smuzhiyun #define DRV_NAME "ide-atapi"
17*4882a593Smuzhiyun #define PFX DRV_NAME ": "
18*4882a593Smuzhiyun
19*4882a593Smuzhiyun #ifdef DEBUG
20*4882a593Smuzhiyun #define debug_log(fmt, args...) \
21*4882a593Smuzhiyun printk(KERN_INFO "ide: " fmt, ## args)
22*4882a593Smuzhiyun #else
23*4882a593Smuzhiyun #define debug_log(fmt, args...) do {} while (0)
24*4882a593Smuzhiyun #endif
25*4882a593Smuzhiyun
26*4882a593Smuzhiyun #define ATAPI_MIN_CDB_BYTES 12
27*4882a593Smuzhiyun
dev_is_idecd(ide_drive_t * drive)28*4882a593Smuzhiyun static inline int dev_is_idecd(ide_drive_t *drive)
29*4882a593Smuzhiyun {
30*4882a593Smuzhiyun return drive->media == ide_cdrom || drive->media == ide_optical;
31*4882a593Smuzhiyun }
32*4882a593Smuzhiyun
33*4882a593Smuzhiyun /*
34*4882a593Smuzhiyun * Check whether we can support a device,
35*4882a593Smuzhiyun * based on the ATAPI IDENTIFY command results.
36*4882a593Smuzhiyun */
ide_check_atapi_device(ide_drive_t * drive,const char * s)37*4882a593Smuzhiyun int ide_check_atapi_device(ide_drive_t *drive, const char *s)
38*4882a593Smuzhiyun {
39*4882a593Smuzhiyun u16 *id = drive->id;
40*4882a593Smuzhiyun u8 gcw[2], protocol, device_type, removable, drq_type, packet_size;
41*4882a593Smuzhiyun
42*4882a593Smuzhiyun *((u16 *)&gcw) = id[ATA_ID_CONFIG];
43*4882a593Smuzhiyun
44*4882a593Smuzhiyun protocol = (gcw[1] & 0xC0) >> 6;
45*4882a593Smuzhiyun device_type = gcw[1] & 0x1F;
46*4882a593Smuzhiyun removable = (gcw[0] & 0x80) >> 7;
47*4882a593Smuzhiyun drq_type = (gcw[0] & 0x60) >> 5;
48*4882a593Smuzhiyun packet_size = gcw[0] & 0x03;
49*4882a593Smuzhiyun
50*4882a593Smuzhiyun #ifdef CONFIG_PPC
51*4882a593Smuzhiyun /* kludge for Apple PowerBook internal zip */
52*4882a593Smuzhiyun if (drive->media == ide_floppy && device_type == 5 &&
53*4882a593Smuzhiyun !strstr((char *)&id[ATA_ID_PROD], "CD-ROM") &&
54*4882a593Smuzhiyun strstr((char *)&id[ATA_ID_PROD], "ZIP"))
55*4882a593Smuzhiyun device_type = 0;
56*4882a593Smuzhiyun #endif
57*4882a593Smuzhiyun
58*4882a593Smuzhiyun if (protocol != 2)
59*4882a593Smuzhiyun printk(KERN_ERR "%s: %s: protocol (0x%02x) is not ATAPI\n",
60*4882a593Smuzhiyun s, drive->name, protocol);
61*4882a593Smuzhiyun else if ((drive->media == ide_floppy && device_type != 0) ||
62*4882a593Smuzhiyun (drive->media == ide_tape && device_type != 1))
63*4882a593Smuzhiyun printk(KERN_ERR "%s: %s: invalid device type (0x%02x)\n",
64*4882a593Smuzhiyun s, drive->name, device_type);
65*4882a593Smuzhiyun else if (removable == 0)
66*4882a593Smuzhiyun printk(KERN_ERR "%s: %s: the removable flag is not set\n",
67*4882a593Smuzhiyun s, drive->name);
68*4882a593Smuzhiyun else if (drive->media == ide_floppy && drq_type == 3)
69*4882a593Smuzhiyun printk(KERN_ERR "%s: %s: sorry, DRQ type (0x%02x) not "
70*4882a593Smuzhiyun "supported\n", s, drive->name, drq_type);
71*4882a593Smuzhiyun else if (packet_size != 0)
72*4882a593Smuzhiyun printk(KERN_ERR "%s: %s: packet size (0x%02x) is not 12 "
73*4882a593Smuzhiyun "bytes\n", s, drive->name, packet_size);
74*4882a593Smuzhiyun else
75*4882a593Smuzhiyun return 1;
76*4882a593Smuzhiyun return 0;
77*4882a593Smuzhiyun }
78*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(ide_check_atapi_device);
79*4882a593Smuzhiyun
ide_init_pc(struct ide_atapi_pc * pc)80*4882a593Smuzhiyun void ide_init_pc(struct ide_atapi_pc *pc)
81*4882a593Smuzhiyun {
82*4882a593Smuzhiyun memset(pc, 0, sizeof(*pc));
83*4882a593Smuzhiyun }
84*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(ide_init_pc);
85*4882a593Smuzhiyun
86*4882a593Smuzhiyun /*
87*4882a593Smuzhiyun * Add a special packet command request to the tail of the request queue,
88*4882a593Smuzhiyun * and wait for it to be serviced.
89*4882a593Smuzhiyun */
ide_queue_pc_tail(ide_drive_t * drive,struct gendisk * disk,struct ide_atapi_pc * pc,void * buf,unsigned int bufflen)90*4882a593Smuzhiyun int ide_queue_pc_tail(ide_drive_t *drive, struct gendisk *disk,
91*4882a593Smuzhiyun struct ide_atapi_pc *pc, void *buf, unsigned int bufflen)
92*4882a593Smuzhiyun {
93*4882a593Smuzhiyun struct request *rq;
94*4882a593Smuzhiyun int error;
95*4882a593Smuzhiyun
96*4882a593Smuzhiyun rq = blk_get_request(drive->queue, REQ_OP_DRV_IN, 0);
97*4882a593Smuzhiyun ide_req(rq)->type = ATA_PRIV_MISC;
98*4882a593Smuzhiyun ide_req(rq)->special = pc;
99*4882a593Smuzhiyun
100*4882a593Smuzhiyun if (buf && bufflen) {
101*4882a593Smuzhiyun error = blk_rq_map_kern(drive->queue, rq, buf, bufflen,
102*4882a593Smuzhiyun GFP_NOIO);
103*4882a593Smuzhiyun if (error)
104*4882a593Smuzhiyun goto put_req;
105*4882a593Smuzhiyun }
106*4882a593Smuzhiyun
107*4882a593Smuzhiyun memcpy(scsi_req(rq)->cmd, pc->c, 12);
108*4882a593Smuzhiyun if (drive->media == ide_tape)
109*4882a593Smuzhiyun scsi_req(rq)->cmd[13] = REQ_IDETAPE_PC1;
110*4882a593Smuzhiyun blk_execute_rq(drive->queue, disk, rq, 0);
111*4882a593Smuzhiyun error = scsi_req(rq)->result ? -EIO : 0;
112*4882a593Smuzhiyun put_req:
113*4882a593Smuzhiyun blk_put_request(rq);
114*4882a593Smuzhiyun return error;
115*4882a593Smuzhiyun }
116*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(ide_queue_pc_tail);
117*4882a593Smuzhiyun
ide_do_test_unit_ready(ide_drive_t * drive,struct gendisk * disk)118*4882a593Smuzhiyun int ide_do_test_unit_ready(ide_drive_t *drive, struct gendisk *disk)
119*4882a593Smuzhiyun {
120*4882a593Smuzhiyun struct ide_atapi_pc pc;
121*4882a593Smuzhiyun
122*4882a593Smuzhiyun ide_init_pc(&pc);
123*4882a593Smuzhiyun pc.c[0] = TEST_UNIT_READY;
124*4882a593Smuzhiyun
125*4882a593Smuzhiyun return ide_queue_pc_tail(drive, disk, &pc, NULL, 0);
126*4882a593Smuzhiyun }
127*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(ide_do_test_unit_ready);
128*4882a593Smuzhiyun
ide_do_start_stop(ide_drive_t * drive,struct gendisk * disk,int start)129*4882a593Smuzhiyun int ide_do_start_stop(ide_drive_t *drive, struct gendisk *disk, int start)
130*4882a593Smuzhiyun {
131*4882a593Smuzhiyun struct ide_atapi_pc pc;
132*4882a593Smuzhiyun
133*4882a593Smuzhiyun ide_init_pc(&pc);
134*4882a593Smuzhiyun pc.c[0] = START_STOP;
135*4882a593Smuzhiyun pc.c[4] = start;
136*4882a593Smuzhiyun
137*4882a593Smuzhiyun if (drive->media == ide_tape)
138*4882a593Smuzhiyun pc.flags |= PC_FLAG_WAIT_FOR_DSC;
139*4882a593Smuzhiyun
140*4882a593Smuzhiyun return ide_queue_pc_tail(drive, disk, &pc, NULL, 0);
141*4882a593Smuzhiyun }
142*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(ide_do_start_stop);
143*4882a593Smuzhiyun
ide_set_media_lock(ide_drive_t * drive,struct gendisk * disk,int on)144*4882a593Smuzhiyun int ide_set_media_lock(ide_drive_t *drive, struct gendisk *disk, int on)
145*4882a593Smuzhiyun {
146*4882a593Smuzhiyun struct ide_atapi_pc pc;
147*4882a593Smuzhiyun
148*4882a593Smuzhiyun if ((drive->dev_flags & IDE_DFLAG_DOORLOCKING) == 0)
149*4882a593Smuzhiyun return 0;
150*4882a593Smuzhiyun
151*4882a593Smuzhiyun ide_init_pc(&pc);
152*4882a593Smuzhiyun pc.c[0] = ALLOW_MEDIUM_REMOVAL;
153*4882a593Smuzhiyun pc.c[4] = on;
154*4882a593Smuzhiyun
155*4882a593Smuzhiyun return ide_queue_pc_tail(drive, disk, &pc, NULL, 0);
156*4882a593Smuzhiyun }
157*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(ide_set_media_lock);
158*4882a593Smuzhiyun
ide_create_request_sense_cmd(ide_drive_t * drive,struct ide_atapi_pc * pc)159*4882a593Smuzhiyun void ide_create_request_sense_cmd(ide_drive_t *drive, struct ide_atapi_pc *pc)
160*4882a593Smuzhiyun {
161*4882a593Smuzhiyun ide_init_pc(pc);
162*4882a593Smuzhiyun pc->c[0] = REQUEST_SENSE;
163*4882a593Smuzhiyun if (drive->media == ide_floppy) {
164*4882a593Smuzhiyun pc->c[4] = 255;
165*4882a593Smuzhiyun pc->req_xfer = 18;
166*4882a593Smuzhiyun } else {
167*4882a593Smuzhiyun pc->c[4] = 20;
168*4882a593Smuzhiyun pc->req_xfer = 20;
169*4882a593Smuzhiyun }
170*4882a593Smuzhiyun }
171*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(ide_create_request_sense_cmd);
172*4882a593Smuzhiyun
ide_prep_sense(ide_drive_t * drive,struct request * rq)173*4882a593Smuzhiyun void ide_prep_sense(ide_drive_t *drive, struct request *rq)
174*4882a593Smuzhiyun {
175*4882a593Smuzhiyun struct request_sense *sense = &drive->sense_data;
176*4882a593Smuzhiyun struct request *sense_rq;
177*4882a593Smuzhiyun struct scsi_request *req;
178*4882a593Smuzhiyun unsigned int cmd_len, sense_len;
179*4882a593Smuzhiyun int err;
180*4882a593Smuzhiyun
181*4882a593Smuzhiyun switch (drive->media) {
182*4882a593Smuzhiyun case ide_floppy:
183*4882a593Smuzhiyun cmd_len = 255;
184*4882a593Smuzhiyun sense_len = 18;
185*4882a593Smuzhiyun break;
186*4882a593Smuzhiyun case ide_tape:
187*4882a593Smuzhiyun cmd_len = 20;
188*4882a593Smuzhiyun sense_len = 20;
189*4882a593Smuzhiyun break;
190*4882a593Smuzhiyun default:
191*4882a593Smuzhiyun cmd_len = 18;
192*4882a593Smuzhiyun sense_len = 18;
193*4882a593Smuzhiyun }
194*4882a593Smuzhiyun
195*4882a593Smuzhiyun BUG_ON(sense_len > sizeof(*sense));
196*4882a593Smuzhiyun
197*4882a593Smuzhiyun if (ata_sense_request(rq) || drive->sense_rq_armed)
198*4882a593Smuzhiyun return;
199*4882a593Smuzhiyun
200*4882a593Smuzhiyun sense_rq = drive->sense_rq;
201*4882a593Smuzhiyun if (!sense_rq) {
202*4882a593Smuzhiyun sense_rq = blk_mq_alloc_request(drive->queue, REQ_OP_DRV_IN,
203*4882a593Smuzhiyun BLK_MQ_REQ_RESERVED | BLK_MQ_REQ_NOWAIT);
204*4882a593Smuzhiyun drive->sense_rq = sense_rq;
205*4882a593Smuzhiyun }
206*4882a593Smuzhiyun req = scsi_req(sense_rq);
207*4882a593Smuzhiyun
208*4882a593Smuzhiyun memset(sense, 0, sizeof(*sense));
209*4882a593Smuzhiyun
210*4882a593Smuzhiyun scsi_req_init(req);
211*4882a593Smuzhiyun
212*4882a593Smuzhiyun err = blk_rq_map_kern(drive->queue, sense_rq, sense, sense_len,
213*4882a593Smuzhiyun GFP_NOIO);
214*4882a593Smuzhiyun if (unlikely(err)) {
215*4882a593Smuzhiyun if (printk_ratelimit())
216*4882a593Smuzhiyun printk(KERN_WARNING PFX "%s: failed to map sense "
217*4882a593Smuzhiyun "buffer\n", drive->name);
218*4882a593Smuzhiyun blk_mq_free_request(sense_rq);
219*4882a593Smuzhiyun drive->sense_rq = NULL;
220*4882a593Smuzhiyun return;
221*4882a593Smuzhiyun }
222*4882a593Smuzhiyun
223*4882a593Smuzhiyun sense_rq->rq_disk = rq->rq_disk;
224*4882a593Smuzhiyun sense_rq->cmd_flags = REQ_OP_DRV_IN;
225*4882a593Smuzhiyun ide_req(sense_rq)->type = ATA_PRIV_SENSE;
226*4882a593Smuzhiyun
227*4882a593Smuzhiyun req->cmd[0] = GPCMD_REQUEST_SENSE;
228*4882a593Smuzhiyun req->cmd[4] = cmd_len;
229*4882a593Smuzhiyun if (drive->media == ide_tape)
230*4882a593Smuzhiyun req->cmd[13] = REQ_IDETAPE_PC1;
231*4882a593Smuzhiyun
232*4882a593Smuzhiyun drive->sense_rq_armed = true;
233*4882a593Smuzhiyun }
234*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(ide_prep_sense);
235*4882a593Smuzhiyun
ide_queue_sense_rq(ide_drive_t * drive,void * special)236*4882a593Smuzhiyun int ide_queue_sense_rq(ide_drive_t *drive, void *special)
237*4882a593Smuzhiyun {
238*4882a593Smuzhiyun ide_hwif_t *hwif = drive->hwif;
239*4882a593Smuzhiyun struct request *sense_rq;
240*4882a593Smuzhiyun unsigned long flags;
241*4882a593Smuzhiyun
242*4882a593Smuzhiyun spin_lock_irqsave(&hwif->lock, flags);
243*4882a593Smuzhiyun
244*4882a593Smuzhiyun /* deferred failure from ide_prep_sense() */
245*4882a593Smuzhiyun if (!drive->sense_rq_armed) {
246*4882a593Smuzhiyun printk(KERN_WARNING PFX "%s: error queuing a sense request\n",
247*4882a593Smuzhiyun drive->name);
248*4882a593Smuzhiyun spin_unlock_irqrestore(&hwif->lock, flags);
249*4882a593Smuzhiyun return -ENOMEM;
250*4882a593Smuzhiyun }
251*4882a593Smuzhiyun
252*4882a593Smuzhiyun sense_rq = drive->sense_rq;
253*4882a593Smuzhiyun ide_req(sense_rq)->special = special;
254*4882a593Smuzhiyun drive->sense_rq_armed = false;
255*4882a593Smuzhiyun
256*4882a593Smuzhiyun drive->hwif->rq = NULL;
257*4882a593Smuzhiyun
258*4882a593Smuzhiyun ide_insert_request_head(drive, sense_rq);
259*4882a593Smuzhiyun spin_unlock_irqrestore(&hwif->lock, flags);
260*4882a593Smuzhiyun return 0;
261*4882a593Smuzhiyun }
262*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(ide_queue_sense_rq);
263*4882a593Smuzhiyun
264*4882a593Smuzhiyun /*
265*4882a593Smuzhiyun * Called when an error was detected during the last packet command.
266*4882a593Smuzhiyun * We queue a request sense packet command at the head of the request
267*4882a593Smuzhiyun * queue.
268*4882a593Smuzhiyun */
ide_retry_pc(ide_drive_t * drive)269*4882a593Smuzhiyun void ide_retry_pc(ide_drive_t *drive)
270*4882a593Smuzhiyun {
271*4882a593Smuzhiyun struct request *failed_rq = drive->hwif->rq;
272*4882a593Smuzhiyun struct request *sense_rq = drive->sense_rq;
273*4882a593Smuzhiyun struct ide_atapi_pc *pc = &drive->request_sense_pc;
274*4882a593Smuzhiyun
275*4882a593Smuzhiyun (void)ide_read_error(drive);
276*4882a593Smuzhiyun
277*4882a593Smuzhiyun /* init pc from sense_rq */
278*4882a593Smuzhiyun ide_init_pc(pc);
279*4882a593Smuzhiyun memcpy(pc->c, scsi_req(sense_rq)->cmd, 12);
280*4882a593Smuzhiyun
281*4882a593Smuzhiyun if (drive->media == ide_tape)
282*4882a593Smuzhiyun drive->atapi_flags |= IDE_AFLAG_IGNORE_DSC;
283*4882a593Smuzhiyun
284*4882a593Smuzhiyun /*
285*4882a593Smuzhiyun * Push back the failed request and put request sense on top
286*4882a593Smuzhiyun * of it. The failed command will be retried after sense data
287*4882a593Smuzhiyun * is acquired.
288*4882a593Smuzhiyun */
289*4882a593Smuzhiyun drive->hwif->rq = NULL;
290*4882a593Smuzhiyun ide_requeue_and_plug(drive, failed_rq);
291*4882a593Smuzhiyun if (ide_queue_sense_rq(drive, pc))
292*4882a593Smuzhiyun ide_complete_rq(drive, BLK_STS_IOERR, blk_rq_bytes(failed_rq));
293*4882a593Smuzhiyun }
294*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(ide_retry_pc);
295*4882a593Smuzhiyun
ide_cd_expiry(ide_drive_t * drive)296*4882a593Smuzhiyun int ide_cd_expiry(ide_drive_t *drive)
297*4882a593Smuzhiyun {
298*4882a593Smuzhiyun struct request *rq = drive->hwif->rq;
299*4882a593Smuzhiyun unsigned long wait = 0;
300*4882a593Smuzhiyun
301*4882a593Smuzhiyun debug_log("%s: scsi_req(rq)->cmd[0]: 0x%x\n", __func__, scsi_req(rq)->cmd[0]);
302*4882a593Smuzhiyun
303*4882a593Smuzhiyun /*
304*4882a593Smuzhiyun * Some commands are *slow* and normally take a long time to complete.
305*4882a593Smuzhiyun * Usually we can use the ATAPI "disconnect" to bypass this, but not all
306*4882a593Smuzhiyun * commands/drives support that. Let ide_timer_expiry keep polling us
307*4882a593Smuzhiyun * for these.
308*4882a593Smuzhiyun */
309*4882a593Smuzhiyun switch (scsi_req(rq)->cmd[0]) {
310*4882a593Smuzhiyun case GPCMD_BLANK:
311*4882a593Smuzhiyun case GPCMD_FORMAT_UNIT:
312*4882a593Smuzhiyun case GPCMD_RESERVE_RZONE_TRACK:
313*4882a593Smuzhiyun case GPCMD_CLOSE_TRACK:
314*4882a593Smuzhiyun case GPCMD_FLUSH_CACHE:
315*4882a593Smuzhiyun wait = ATAPI_WAIT_PC;
316*4882a593Smuzhiyun break;
317*4882a593Smuzhiyun default:
318*4882a593Smuzhiyun if (!(rq->rq_flags & RQF_QUIET))
319*4882a593Smuzhiyun printk(KERN_INFO PFX "cmd 0x%x timed out\n",
320*4882a593Smuzhiyun scsi_req(rq)->cmd[0]);
321*4882a593Smuzhiyun wait = 0;
322*4882a593Smuzhiyun break;
323*4882a593Smuzhiyun }
324*4882a593Smuzhiyun return wait;
325*4882a593Smuzhiyun }
326*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(ide_cd_expiry);
327*4882a593Smuzhiyun
ide_cd_get_xferlen(struct request * rq)328*4882a593Smuzhiyun int ide_cd_get_xferlen(struct request *rq)
329*4882a593Smuzhiyun {
330*4882a593Smuzhiyun switch (req_op(rq)) {
331*4882a593Smuzhiyun default:
332*4882a593Smuzhiyun return 32768;
333*4882a593Smuzhiyun case REQ_OP_SCSI_IN:
334*4882a593Smuzhiyun case REQ_OP_SCSI_OUT:
335*4882a593Smuzhiyun return blk_rq_bytes(rq);
336*4882a593Smuzhiyun case REQ_OP_DRV_IN:
337*4882a593Smuzhiyun case REQ_OP_DRV_OUT:
338*4882a593Smuzhiyun switch (ide_req(rq)->type) {
339*4882a593Smuzhiyun case ATA_PRIV_PC:
340*4882a593Smuzhiyun case ATA_PRIV_SENSE:
341*4882a593Smuzhiyun return blk_rq_bytes(rq);
342*4882a593Smuzhiyun default:
343*4882a593Smuzhiyun return 0;
344*4882a593Smuzhiyun }
345*4882a593Smuzhiyun }
346*4882a593Smuzhiyun }
347*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(ide_cd_get_xferlen);
348*4882a593Smuzhiyun
ide_read_bcount_and_ireason(ide_drive_t * drive,u16 * bcount,u8 * ireason)349*4882a593Smuzhiyun void ide_read_bcount_and_ireason(ide_drive_t *drive, u16 *bcount, u8 *ireason)
350*4882a593Smuzhiyun {
351*4882a593Smuzhiyun struct ide_taskfile tf;
352*4882a593Smuzhiyun
353*4882a593Smuzhiyun drive->hwif->tp_ops->tf_read(drive, &tf, IDE_VALID_NSECT |
354*4882a593Smuzhiyun IDE_VALID_LBAM | IDE_VALID_LBAH);
355*4882a593Smuzhiyun
356*4882a593Smuzhiyun *bcount = (tf.lbah << 8) | tf.lbam;
357*4882a593Smuzhiyun *ireason = tf.nsect & 3;
358*4882a593Smuzhiyun }
359*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(ide_read_bcount_and_ireason);
360*4882a593Smuzhiyun
361*4882a593Smuzhiyun /*
362*4882a593Smuzhiyun * Check the contents of the interrupt reason register and attempt to recover if
363*4882a593Smuzhiyun * there are problems.
364*4882a593Smuzhiyun *
365*4882a593Smuzhiyun * Returns:
366*4882a593Smuzhiyun * - 0 if everything's ok
367*4882a593Smuzhiyun * - 1 if the request has to be terminated.
368*4882a593Smuzhiyun */
ide_check_ireason(ide_drive_t * drive,struct request * rq,int len,int ireason,int rw)369*4882a593Smuzhiyun int ide_check_ireason(ide_drive_t *drive, struct request *rq, int len,
370*4882a593Smuzhiyun int ireason, int rw)
371*4882a593Smuzhiyun {
372*4882a593Smuzhiyun ide_hwif_t *hwif = drive->hwif;
373*4882a593Smuzhiyun
374*4882a593Smuzhiyun debug_log("ireason: 0x%x, rw: 0x%x\n", ireason, rw);
375*4882a593Smuzhiyun
376*4882a593Smuzhiyun if (ireason == (!rw << 1))
377*4882a593Smuzhiyun return 0;
378*4882a593Smuzhiyun else if (ireason == (rw << 1)) {
379*4882a593Smuzhiyun printk(KERN_ERR PFX "%s: %s: wrong transfer direction!\n",
380*4882a593Smuzhiyun drive->name, __func__);
381*4882a593Smuzhiyun
382*4882a593Smuzhiyun if (dev_is_idecd(drive))
383*4882a593Smuzhiyun ide_pad_transfer(drive, rw, len);
384*4882a593Smuzhiyun } else if (!rw && ireason == ATAPI_COD) {
385*4882a593Smuzhiyun if (dev_is_idecd(drive)) {
386*4882a593Smuzhiyun /*
387*4882a593Smuzhiyun * Some drives (ASUS) seem to tell us that status info
388*4882a593Smuzhiyun * is available. Just get it and ignore.
389*4882a593Smuzhiyun */
390*4882a593Smuzhiyun (void)hwif->tp_ops->read_status(hwif);
391*4882a593Smuzhiyun return 0;
392*4882a593Smuzhiyun }
393*4882a593Smuzhiyun } else {
394*4882a593Smuzhiyun if (ireason & ATAPI_COD)
395*4882a593Smuzhiyun printk(KERN_ERR PFX "%s: CoD != 0 in %s\n", drive->name,
396*4882a593Smuzhiyun __func__);
397*4882a593Smuzhiyun
398*4882a593Smuzhiyun /* drive wants a command packet, or invalid ireason... */
399*4882a593Smuzhiyun printk(KERN_ERR PFX "%s: %s: bad interrupt reason 0x%02x\n",
400*4882a593Smuzhiyun drive->name, __func__, ireason);
401*4882a593Smuzhiyun }
402*4882a593Smuzhiyun
403*4882a593Smuzhiyun if (dev_is_idecd(drive) && ata_pc_request(rq))
404*4882a593Smuzhiyun rq->rq_flags |= RQF_FAILED;
405*4882a593Smuzhiyun
406*4882a593Smuzhiyun return 1;
407*4882a593Smuzhiyun }
408*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(ide_check_ireason);
409*4882a593Smuzhiyun
410*4882a593Smuzhiyun /*
411*4882a593Smuzhiyun * This is the usual interrupt handler which will be called during a packet
412*4882a593Smuzhiyun * command. We will transfer some of the data (as requested by the drive)
413*4882a593Smuzhiyun * and will re-point interrupt handler to us.
414*4882a593Smuzhiyun */
ide_pc_intr(ide_drive_t * drive)415*4882a593Smuzhiyun static ide_startstop_t ide_pc_intr(ide_drive_t *drive)
416*4882a593Smuzhiyun {
417*4882a593Smuzhiyun struct ide_atapi_pc *pc = drive->pc;
418*4882a593Smuzhiyun ide_hwif_t *hwif = drive->hwif;
419*4882a593Smuzhiyun struct ide_cmd *cmd = &hwif->cmd;
420*4882a593Smuzhiyun struct request *rq = hwif->rq;
421*4882a593Smuzhiyun const struct ide_tp_ops *tp_ops = hwif->tp_ops;
422*4882a593Smuzhiyun unsigned int timeout, done;
423*4882a593Smuzhiyun u16 bcount;
424*4882a593Smuzhiyun u8 stat, ireason, dsc = 0;
425*4882a593Smuzhiyun u8 write = !!(pc->flags & PC_FLAG_WRITING);
426*4882a593Smuzhiyun
427*4882a593Smuzhiyun debug_log("Enter %s - interrupt handler\n", __func__);
428*4882a593Smuzhiyun
429*4882a593Smuzhiyun timeout = (drive->media == ide_floppy) ? WAIT_FLOPPY_CMD
430*4882a593Smuzhiyun : WAIT_TAPE_CMD;
431*4882a593Smuzhiyun
432*4882a593Smuzhiyun /* Clear the interrupt */
433*4882a593Smuzhiyun stat = tp_ops->read_status(hwif);
434*4882a593Smuzhiyun
435*4882a593Smuzhiyun if (pc->flags & PC_FLAG_DMA_IN_PROGRESS) {
436*4882a593Smuzhiyun int rc;
437*4882a593Smuzhiyun
438*4882a593Smuzhiyun drive->waiting_for_dma = 0;
439*4882a593Smuzhiyun rc = hwif->dma_ops->dma_end(drive);
440*4882a593Smuzhiyun ide_dma_unmap_sg(drive, cmd);
441*4882a593Smuzhiyun
442*4882a593Smuzhiyun if (rc || (drive->media == ide_tape && (stat & ATA_ERR))) {
443*4882a593Smuzhiyun if (drive->media == ide_floppy)
444*4882a593Smuzhiyun printk(KERN_ERR PFX "%s: DMA %s error\n",
445*4882a593Smuzhiyun drive->name, rq_data_dir(pc->rq)
446*4882a593Smuzhiyun ? "write" : "read");
447*4882a593Smuzhiyun pc->flags |= PC_FLAG_DMA_ERROR;
448*4882a593Smuzhiyun } else
449*4882a593Smuzhiyun scsi_req(rq)->resid_len = 0;
450*4882a593Smuzhiyun debug_log("%s: DMA finished\n", drive->name);
451*4882a593Smuzhiyun }
452*4882a593Smuzhiyun
453*4882a593Smuzhiyun /* No more interrupts */
454*4882a593Smuzhiyun if ((stat & ATA_DRQ) == 0) {
455*4882a593Smuzhiyun int uptodate;
456*4882a593Smuzhiyun blk_status_t error;
457*4882a593Smuzhiyun
458*4882a593Smuzhiyun debug_log("Packet command completed, %d bytes transferred\n",
459*4882a593Smuzhiyun blk_rq_bytes(rq));
460*4882a593Smuzhiyun
461*4882a593Smuzhiyun pc->flags &= ~PC_FLAG_DMA_IN_PROGRESS;
462*4882a593Smuzhiyun
463*4882a593Smuzhiyun local_irq_enable_in_hardirq();
464*4882a593Smuzhiyun
465*4882a593Smuzhiyun if (drive->media == ide_tape &&
466*4882a593Smuzhiyun (stat & ATA_ERR) && scsi_req(rq)->cmd[0] == REQUEST_SENSE)
467*4882a593Smuzhiyun stat &= ~ATA_ERR;
468*4882a593Smuzhiyun
469*4882a593Smuzhiyun if ((stat & ATA_ERR) || (pc->flags & PC_FLAG_DMA_ERROR)) {
470*4882a593Smuzhiyun /* Error detected */
471*4882a593Smuzhiyun debug_log("%s: I/O error\n", drive->name);
472*4882a593Smuzhiyun
473*4882a593Smuzhiyun if (drive->media != ide_tape)
474*4882a593Smuzhiyun scsi_req(pc->rq)->result++;
475*4882a593Smuzhiyun
476*4882a593Smuzhiyun if (scsi_req(rq)->cmd[0] == REQUEST_SENSE) {
477*4882a593Smuzhiyun printk(KERN_ERR PFX "%s: I/O error in request "
478*4882a593Smuzhiyun "sense command\n", drive->name);
479*4882a593Smuzhiyun return ide_do_reset(drive);
480*4882a593Smuzhiyun }
481*4882a593Smuzhiyun
482*4882a593Smuzhiyun debug_log("[cmd %x]: check condition\n", scsi_req(rq)->cmd[0]);
483*4882a593Smuzhiyun
484*4882a593Smuzhiyun /* Retry operation */
485*4882a593Smuzhiyun ide_retry_pc(drive);
486*4882a593Smuzhiyun
487*4882a593Smuzhiyun /* queued, but not started */
488*4882a593Smuzhiyun return ide_stopped;
489*4882a593Smuzhiyun }
490*4882a593Smuzhiyun pc->error = 0;
491*4882a593Smuzhiyun
492*4882a593Smuzhiyun if ((pc->flags & PC_FLAG_WAIT_FOR_DSC) && (stat & ATA_DSC) == 0)
493*4882a593Smuzhiyun dsc = 1;
494*4882a593Smuzhiyun
495*4882a593Smuzhiyun /*
496*4882a593Smuzhiyun * ->pc_callback() might change rq->data_len for
497*4882a593Smuzhiyun * residual count, cache total length.
498*4882a593Smuzhiyun */
499*4882a593Smuzhiyun done = blk_rq_bytes(rq);
500*4882a593Smuzhiyun
501*4882a593Smuzhiyun /* Command finished - Call the callback function */
502*4882a593Smuzhiyun uptodate = drive->pc_callback(drive, dsc);
503*4882a593Smuzhiyun
504*4882a593Smuzhiyun if (uptodate == 0)
505*4882a593Smuzhiyun drive->failed_pc = NULL;
506*4882a593Smuzhiyun
507*4882a593Smuzhiyun if (ata_misc_request(rq)) {
508*4882a593Smuzhiyun scsi_req(rq)->result = 0;
509*4882a593Smuzhiyun error = BLK_STS_OK;
510*4882a593Smuzhiyun } else {
511*4882a593Smuzhiyun
512*4882a593Smuzhiyun if (blk_rq_is_passthrough(rq) && uptodate <= 0) {
513*4882a593Smuzhiyun if (scsi_req(rq)->result == 0)
514*4882a593Smuzhiyun scsi_req(rq)->result = -EIO;
515*4882a593Smuzhiyun }
516*4882a593Smuzhiyun
517*4882a593Smuzhiyun error = uptodate ? BLK_STS_OK : BLK_STS_IOERR;
518*4882a593Smuzhiyun }
519*4882a593Smuzhiyun
520*4882a593Smuzhiyun ide_complete_rq(drive, error, blk_rq_bytes(rq));
521*4882a593Smuzhiyun return ide_stopped;
522*4882a593Smuzhiyun }
523*4882a593Smuzhiyun
524*4882a593Smuzhiyun if (pc->flags & PC_FLAG_DMA_IN_PROGRESS) {
525*4882a593Smuzhiyun pc->flags &= ~PC_FLAG_DMA_IN_PROGRESS;
526*4882a593Smuzhiyun printk(KERN_ERR PFX "%s: The device wants to issue more "
527*4882a593Smuzhiyun "interrupts in DMA mode\n", drive->name);
528*4882a593Smuzhiyun ide_dma_off(drive);
529*4882a593Smuzhiyun return ide_do_reset(drive);
530*4882a593Smuzhiyun }
531*4882a593Smuzhiyun
532*4882a593Smuzhiyun /* Get the number of bytes to transfer on this interrupt. */
533*4882a593Smuzhiyun ide_read_bcount_and_ireason(drive, &bcount, &ireason);
534*4882a593Smuzhiyun
535*4882a593Smuzhiyun if (ide_check_ireason(drive, rq, bcount, ireason, write))
536*4882a593Smuzhiyun return ide_do_reset(drive);
537*4882a593Smuzhiyun
538*4882a593Smuzhiyun done = min_t(unsigned int, bcount, cmd->nleft);
539*4882a593Smuzhiyun ide_pio_bytes(drive, cmd, write, done);
540*4882a593Smuzhiyun
541*4882a593Smuzhiyun /* Update transferred byte count */
542*4882a593Smuzhiyun scsi_req(rq)->resid_len -= done;
543*4882a593Smuzhiyun
544*4882a593Smuzhiyun bcount -= done;
545*4882a593Smuzhiyun
546*4882a593Smuzhiyun if (bcount)
547*4882a593Smuzhiyun ide_pad_transfer(drive, write, bcount);
548*4882a593Smuzhiyun
549*4882a593Smuzhiyun debug_log("[cmd %x] transferred %d bytes, padded %d bytes, resid: %u\n",
550*4882a593Smuzhiyun scsi_req(rq)->cmd[0], done, bcount, scsi_req(rq)->resid_len);
551*4882a593Smuzhiyun
552*4882a593Smuzhiyun /* And set the interrupt handler again */
553*4882a593Smuzhiyun ide_set_handler(drive, ide_pc_intr, timeout);
554*4882a593Smuzhiyun return ide_started;
555*4882a593Smuzhiyun }
556*4882a593Smuzhiyun
ide_init_packet_cmd(struct ide_cmd * cmd,u8 valid_tf,u16 bcount,u8 dma)557*4882a593Smuzhiyun static void ide_init_packet_cmd(struct ide_cmd *cmd, u8 valid_tf,
558*4882a593Smuzhiyun u16 bcount, u8 dma)
559*4882a593Smuzhiyun {
560*4882a593Smuzhiyun cmd->protocol = dma ? ATAPI_PROT_DMA : ATAPI_PROT_PIO;
561*4882a593Smuzhiyun cmd->valid.out.tf = IDE_VALID_LBAH | IDE_VALID_LBAM |
562*4882a593Smuzhiyun IDE_VALID_FEATURE | valid_tf;
563*4882a593Smuzhiyun cmd->tf.command = ATA_CMD_PACKET;
564*4882a593Smuzhiyun cmd->tf.feature = dma; /* Use PIO/DMA */
565*4882a593Smuzhiyun cmd->tf.lbam = bcount & 0xff;
566*4882a593Smuzhiyun cmd->tf.lbah = (bcount >> 8) & 0xff;
567*4882a593Smuzhiyun }
568*4882a593Smuzhiyun
ide_read_ireason(ide_drive_t * drive)569*4882a593Smuzhiyun static u8 ide_read_ireason(ide_drive_t *drive)
570*4882a593Smuzhiyun {
571*4882a593Smuzhiyun struct ide_taskfile tf;
572*4882a593Smuzhiyun
573*4882a593Smuzhiyun drive->hwif->tp_ops->tf_read(drive, &tf, IDE_VALID_NSECT);
574*4882a593Smuzhiyun
575*4882a593Smuzhiyun return tf.nsect & 3;
576*4882a593Smuzhiyun }
577*4882a593Smuzhiyun
ide_wait_ireason(ide_drive_t * drive,u8 ireason)578*4882a593Smuzhiyun static u8 ide_wait_ireason(ide_drive_t *drive, u8 ireason)
579*4882a593Smuzhiyun {
580*4882a593Smuzhiyun int retries = 100;
581*4882a593Smuzhiyun
582*4882a593Smuzhiyun while (retries-- && ((ireason & ATAPI_COD) == 0 ||
583*4882a593Smuzhiyun (ireason & ATAPI_IO))) {
584*4882a593Smuzhiyun printk(KERN_ERR PFX "%s: (IO,CoD != (0,1) while issuing "
585*4882a593Smuzhiyun "a packet command, retrying\n", drive->name);
586*4882a593Smuzhiyun udelay(100);
587*4882a593Smuzhiyun ireason = ide_read_ireason(drive);
588*4882a593Smuzhiyun if (retries == 0) {
589*4882a593Smuzhiyun printk(KERN_ERR PFX "%s: (IO,CoD != (0,1) while issuing"
590*4882a593Smuzhiyun " a packet command, ignoring\n",
591*4882a593Smuzhiyun drive->name);
592*4882a593Smuzhiyun ireason |= ATAPI_COD;
593*4882a593Smuzhiyun ireason &= ~ATAPI_IO;
594*4882a593Smuzhiyun }
595*4882a593Smuzhiyun }
596*4882a593Smuzhiyun
597*4882a593Smuzhiyun return ireason;
598*4882a593Smuzhiyun }
599*4882a593Smuzhiyun
ide_delayed_transfer_pc(ide_drive_t * drive)600*4882a593Smuzhiyun static int ide_delayed_transfer_pc(ide_drive_t *drive)
601*4882a593Smuzhiyun {
602*4882a593Smuzhiyun /* Send the actual packet */
603*4882a593Smuzhiyun drive->hwif->tp_ops->output_data(drive, NULL, drive->pc->c, 12);
604*4882a593Smuzhiyun
605*4882a593Smuzhiyun /* Timeout for the packet command */
606*4882a593Smuzhiyun return WAIT_FLOPPY_CMD;
607*4882a593Smuzhiyun }
608*4882a593Smuzhiyun
ide_transfer_pc(ide_drive_t * drive)609*4882a593Smuzhiyun static ide_startstop_t ide_transfer_pc(ide_drive_t *drive)
610*4882a593Smuzhiyun {
611*4882a593Smuzhiyun struct ide_atapi_pc *pc;
612*4882a593Smuzhiyun ide_hwif_t *hwif = drive->hwif;
613*4882a593Smuzhiyun struct request *rq = hwif->rq;
614*4882a593Smuzhiyun ide_expiry_t *expiry;
615*4882a593Smuzhiyun unsigned int timeout;
616*4882a593Smuzhiyun int cmd_len;
617*4882a593Smuzhiyun ide_startstop_t startstop;
618*4882a593Smuzhiyun u8 ireason;
619*4882a593Smuzhiyun
620*4882a593Smuzhiyun if (ide_wait_stat(&startstop, drive, ATA_DRQ, ATA_BUSY, WAIT_READY)) {
621*4882a593Smuzhiyun printk(KERN_ERR PFX "%s: Strange, packet command initiated yet "
622*4882a593Smuzhiyun "DRQ isn't asserted\n", drive->name);
623*4882a593Smuzhiyun return startstop;
624*4882a593Smuzhiyun }
625*4882a593Smuzhiyun
626*4882a593Smuzhiyun if (drive->atapi_flags & IDE_AFLAG_DRQ_INTERRUPT) {
627*4882a593Smuzhiyun if (drive->dma)
628*4882a593Smuzhiyun drive->waiting_for_dma = 1;
629*4882a593Smuzhiyun }
630*4882a593Smuzhiyun
631*4882a593Smuzhiyun if (dev_is_idecd(drive)) {
632*4882a593Smuzhiyun /* ATAPI commands get padded out to 12 bytes minimum */
633*4882a593Smuzhiyun cmd_len = COMMAND_SIZE(scsi_req(rq)->cmd[0]);
634*4882a593Smuzhiyun if (cmd_len < ATAPI_MIN_CDB_BYTES)
635*4882a593Smuzhiyun cmd_len = ATAPI_MIN_CDB_BYTES;
636*4882a593Smuzhiyun
637*4882a593Smuzhiyun timeout = rq->timeout;
638*4882a593Smuzhiyun expiry = ide_cd_expiry;
639*4882a593Smuzhiyun } else {
640*4882a593Smuzhiyun pc = drive->pc;
641*4882a593Smuzhiyun
642*4882a593Smuzhiyun cmd_len = ATAPI_MIN_CDB_BYTES;
643*4882a593Smuzhiyun
644*4882a593Smuzhiyun /*
645*4882a593Smuzhiyun * If necessary schedule the packet transfer to occur 'timeout'
646*4882a593Smuzhiyun * milliseconds later in ide_delayed_transfer_pc() after the
647*4882a593Smuzhiyun * device says it's ready for a packet.
648*4882a593Smuzhiyun */
649*4882a593Smuzhiyun if (drive->atapi_flags & IDE_AFLAG_ZIP_DRIVE) {
650*4882a593Smuzhiyun timeout = drive->pc_delay;
651*4882a593Smuzhiyun expiry = &ide_delayed_transfer_pc;
652*4882a593Smuzhiyun } else {
653*4882a593Smuzhiyun timeout = (drive->media == ide_floppy) ? WAIT_FLOPPY_CMD
654*4882a593Smuzhiyun : WAIT_TAPE_CMD;
655*4882a593Smuzhiyun expiry = NULL;
656*4882a593Smuzhiyun }
657*4882a593Smuzhiyun
658*4882a593Smuzhiyun ireason = ide_read_ireason(drive);
659*4882a593Smuzhiyun if (drive->media == ide_tape)
660*4882a593Smuzhiyun ireason = ide_wait_ireason(drive, ireason);
661*4882a593Smuzhiyun
662*4882a593Smuzhiyun if ((ireason & ATAPI_COD) == 0 || (ireason & ATAPI_IO)) {
663*4882a593Smuzhiyun printk(KERN_ERR PFX "%s: (IO,CoD) != (0,1) while "
664*4882a593Smuzhiyun "issuing a packet command\n", drive->name);
665*4882a593Smuzhiyun
666*4882a593Smuzhiyun return ide_do_reset(drive);
667*4882a593Smuzhiyun }
668*4882a593Smuzhiyun }
669*4882a593Smuzhiyun
670*4882a593Smuzhiyun hwif->expiry = expiry;
671*4882a593Smuzhiyun
672*4882a593Smuzhiyun /* Set the interrupt routine */
673*4882a593Smuzhiyun ide_set_handler(drive,
674*4882a593Smuzhiyun (dev_is_idecd(drive) ? drive->irq_handler
675*4882a593Smuzhiyun : ide_pc_intr),
676*4882a593Smuzhiyun timeout);
677*4882a593Smuzhiyun
678*4882a593Smuzhiyun /* Send the actual packet */
679*4882a593Smuzhiyun if ((drive->atapi_flags & IDE_AFLAG_ZIP_DRIVE) == 0)
680*4882a593Smuzhiyun hwif->tp_ops->output_data(drive, NULL, scsi_req(rq)->cmd, cmd_len);
681*4882a593Smuzhiyun
682*4882a593Smuzhiyun /* Begin DMA, if necessary */
683*4882a593Smuzhiyun if (dev_is_idecd(drive)) {
684*4882a593Smuzhiyun if (drive->dma)
685*4882a593Smuzhiyun hwif->dma_ops->dma_start(drive);
686*4882a593Smuzhiyun } else {
687*4882a593Smuzhiyun if (pc->flags & PC_FLAG_DMA_OK) {
688*4882a593Smuzhiyun pc->flags |= PC_FLAG_DMA_IN_PROGRESS;
689*4882a593Smuzhiyun hwif->dma_ops->dma_start(drive);
690*4882a593Smuzhiyun }
691*4882a593Smuzhiyun }
692*4882a593Smuzhiyun
693*4882a593Smuzhiyun return ide_started;
694*4882a593Smuzhiyun }
695*4882a593Smuzhiyun
ide_issue_pc(ide_drive_t * drive,struct ide_cmd * cmd)696*4882a593Smuzhiyun ide_startstop_t ide_issue_pc(ide_drive_t *drive, struct ide_cmd *cmd)
697*4882a593Smuzhiyun {
698*4882a593Smuzhiyun struct ide_atapi_pc *pc;
699*4882a593Smuzhiyun ide_hwif_t *hwif = drive->hwif;
700*4882a593Smuzhiyun ide_expiry_t *expiry = NULL;
701*4882a593Smuzhiyun struct request *rq = hwif->rq;
702*4882a593Smuzhiyun unsigned int timeout, bytes;
703*4882a593Smuzhiyun u16 bcount;
704*4882a593Smuzhiyun u8 valid_tf;
705*4882a593Smuzhiyun u8 drq_int = !!(drive->atapi_flags & IDE_AFLAG_DRQ_INTERRUPT);
706*4882a593Smuzhiyun
707*4882a593Smuzhiyun if (dev_is_idecd(drive)) {
708*4882a593Smuzhiyun valid_tf = IDE_VALID_NSECT | IDE_VALID_LBAL;
709*4882a593Smuzhiyun bcount = ide_cd_get_xferlen(rq);
710*4882a593Smuzhiyun expiry = ide_cd_expiry;
711*4882a593Smuzhiyun timeout = ATAPI_WAIT_PC;
712*4882a593Smuzhiyun
713*4882a593Smuzhiyun if (drive->dma)
714*4882a593Smuzhiyun drive->dma = !ide_dma_prepare(drive, cmd);
715*4882a593Smuzhiyun } else {
716*4882a593Smuzhiyun pc = drive->pc;
717*4882a593Smuzhiyun
718*4882a593Smuzhiyun valid_tf = IDE_VALID_DEVICE;
719*4882a593Smuzhiyun bytes = blk_rq_bytes(rq);
720*4882a593Smuzhiyun bcount = ((drive->media == ide_tape) ? bytes
721*4882a593Smuzhiyun : min_t(unsigned int,
722*4882a593Smuzhiyun bytes, 63 * 1024));
723*4882a593Smuzhiyun
724*4882a593Smuzhiyun /* We haven't transferred any data yet */
725*4882a593Smuzhiyun scsi_req(rq)->resid_len = bcount;
726*4882a593Smuzhiyun
727*4882a593Smuzhiyun if (pc->flags & PC_FLAG_DMA_ERROR) {
728*4882a593Smuzhiyun pc->flags &= ~PC_FLAG_DMA_ERROR;
729*4882a593Smuzhiyun ide_dma_off(drive);
730*4882a593Smuzhiyun }
731*4882a593Smuzhiyun
732*4882a593Smuzhiyun if (pc->flags & PC_FLAG_DMA_OK)
733*4882a593Smuzhiyun drive->dma = !ide_dma_prepare(drive, cmd);
734*4882a593Smuzhiyun
735*4882a593Smuzhiyun if (!drive->dma)
736*4882a593Smuzhiyun pc->flags &= ~PC_FLAG_DMA_OK;
737*4882a593Smuzhiyun
738*4882a593Smuzhiyun timeout = (drive->media == ide_floppy) ? WAIT_FLOPPY_CMD
739*4882a593Smuzhiyun : WAIT_TAPE_CMD;
740*4882a593Smuzhiyun }
741*4882a593Smuzhiyun
742*4882a593Smuzhiyun ide_init_packet_cmd(cmd, valid_tf, bcount, drive->dma);
743*4882a593Smuzhiyun
744*4882a593Smuzhiyun (void)do_rw_taskfile(drive, cmd);
745*4882a593Smuzhiyun
746*4882a593Smuzhiyun if (drq_int) {
747*4882a593Smuzhiyun if (drive->dma)
748*4882a593Smuzhiyun drive->waiting_for_dma = 0;
749*4882a593Smuzhiyun hwif->expiry = expiry;
750*4882a593Smuzhiyun }
751*4882a593Smuzhiyun
752*4882a593Smuzhiyun ide_execute_command(drive, cmd, ide_transfer_pc, timeout);
753*4882a593Smuzhiyun
754*4882a593Smuzhiyun return drq_int ? ide_started : ide_transfer_pc(drive);
755*4882a593Smuzhiyun }
756*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(ide_issue_pc);
757