Subversion Repositories shark

Rev

Rev 422 | Details | Compare with Previous | Last modification | View Log | RSS feed

Rev Author Line No. Line
422 giacomo 1
/**** vi:set ts=8 sts=8 sw=8:************************************************
2
 *
3
 * Copyright (C) 2002 Marcin Dalecki <martin@dalecki.de>
4
 *
5
 * This program is free software; you can redistribute it and/or modify it
6
 * under the terms of the GNU General Public License version 2 as published by
7
 * the Free Software Foundation.
8
 *
9
 * This program is distributed in the hope that it will be useful, but WITHOUT
10
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for
12
 * more details.
13
 */
14
 
15
#include <linux/types.h>
16
#include <asm/byteorder.h>
17
 
18
/*
19
 * With each packet command, we allocate a buffer.
20
 * This is used for several packet
21
 * commands (Not for READ/WRITE commands).
22
 */
23
#define IDEFLOPPY_PC_BUFFER_SIZE        256
24
#define IDETAPE_PC_BUFFER_SIZE          256
25
 
26
/*
27
 * Packet flags bits.
28
 */
29
 
30
#define PC_ABORT                0       /* set when an error is considered normal - we won't retry */
31
#define PC_WAIT_FOR_DSC         1       /* 1 when polling for DSC on a media access command */
32
#define PC_DMA_RECOMMENDED      2       /* 1 when we prefer to use DMA if possible */
33
#define PC_DMA_IN_PROGRESS      3       /* 1 while DMA in progress */
34
#define PC_DMA_ERROR            4       /* 1 when encountered problem during DMA */
35
#define PC_WRITING              5       /* data direction */
36
#define PC_SUPPRESS_ERROR       6       /* suppress error reporting */
37
#define PC_TRANSFORM            7       /* transform SCSI commands */
38
 
39
/* This struct get's shared between different drivers.
40
 */
41
struct atapi_packet_command {
42
        u8 c[12];                       /* Actual packet bytes */
43
        char *buffer;                   /* Data buffer */
44
        int buffer_size;                /* Size of our data buffer */
45
        char *current_position;         /* Pointer into the above buffer */
46
        int request_transfer;           /* Bytes to transfer */
47
        int actually_transferred;       /* Bytes actually transferred */
48
 
49
        unsigned long flags;            /* Status/Action bit flags: long for set_bit */
50
 
51
        /* FIXME: the following is ugly as hell, but the only way we can start
52
         * actually to unify the code.
53
         */
54
        /* driver specific data. */
55
        /* floppy/tape */
56
        int retries;                            /* On each retry, we increment retries */
57
        int error;                              /* Error code */
58
        char *b_data;                           /* Pointer which runs on the buffers */
59
        unsigned int b_count;                   /* Missing/Available data on the current buffer */
60
        u8 pc_buffer[IDEFLOPPY_PC_BUFFER_SIZE]; /* Temporary buffer */
61
        /* Called when this packet command is completed */
62
        void (*callback) (struct ata_device *, struct request *);
63
 
64
        /* only tape */
65
        struct bio *bio;
66
 
67
        /* only scsi */
68
        struct {
69
                unsigned int b_count;                   /* Bytes transferred from current entry */
70
                struct scatterlist *sg;                 /* Scatter gather table */
71
                struct scsi_cmnd *scsi_cmd;             /* SCSI command */
72
                void (*done)(struct scsi_cmnd *);       /* Scsi completion routine */
73
                unsigned long timeout;                  /* Command timeout */
74
        } s;
75
};
76
 
77
/*
78
 *      ATAPI Status Register.
79
 */
80
typedef union {
81
        u8 all                  : 8;
82
        struct {
83
#if defined(__LITTLE_ENDIAN_BITFIELD)
84
                u8 check        : 1;    /* Error occurred */
85
                u8 idx          : 1;    /* Reserved */
86
                u8 corr         : 1;    /* Correctable error occurred */
87
                u8 drq          : 1;    /* Data is request by the device */
88
                u8 dsc          : 1;    /* Media access command finished / Buffer availability */
89
                u8 reserved5    : 1;    /* Reserved */
90
                u8 drdy         : 1;    /* Ignored for ATAPI commands (ready to accept ATA command) */
91
                u8 bsy          : 1;    /* The device has access to the command block */
92
#elif defined(__BIG_ENDIAN_BITFIELD)
93
                u8 bsy          : 1;
94
                u8 drdy         : 1;
95
                u8 reserved5    : 1;
96
                u8 dsc          : 1;
97
                u8 drq          : 1;
98
                u8 corr         : 1;
99
                u8 idx          : 1;
100
                u8 check        : 1;
101
#else
102
#error "Please fix <asm/byteorder.h>"
103
#endif
104
        } b;
105
} atapi_status_reg_t;
106
 
107
/*
108
 *      ATAPI error register.
109
 */
110
typedef union {
111
        u8 all                  : 8;
112
        struct {
113
#if defined(__LITTLE_ENDIAN_BITFIELD)
114
                u8 ili          : 1;    /* Illegal Length Indication */
115
                u8 eom          : 1;    /* End Of Media Detected */
116
                u8 abrt         : 1;    /* Aborted command - As defined by ATA */
117
                u8 mcr          : 1;    /* Media Change Requested - As defined by ATA */
118
                u8 sense_key    : 4;    /* Sense key of the last failed packet command */
119
#elif defined(__BIG_ENDIAN_BITFIELD)
120
                u8 sense_key    : 4;
121
                u8 mcr          : 1;
122
                u8 abrt         : 1;
123
                u8 eom          : 1;
124
                u8 ili          : 1;
125
#else
126
#error "Please fix <asm/byteorder.h>"
127
#endif
128
        } b;
129
} atapi_error_reg_t;
130
 
131
/* Currently unused, but please do not remove.  --bkz */
132
/*
133
 *      ATAPI Feature Register.
134
 */
135
typedef union {
136
        u8 all                  : 8;
137
        struct {
138
#if defined(__LITTLE_ENDIAN_BITFIELD)
139
                u8 dma          : 1;    /* Using DMA or PIO */
140
                u8 reserved321  : 3;    /* Reserved */
141
                u8 reserved654  : 3;    /* Reserved (Tag Type) */
142
                u8 reserved7    : 1;    /* Reserved */
143
#elif defined(__BIG_ENDIAN_BITFIELD)
144
                u8 reserved7    : 1;
145
                u8 reserved654  : 3;
146
                u8 reserved321  : 3;
147
                u8 dma          : 1;
148
#else
149
#error "Please fix <asm/byteorder.h>"
150
#endif
151
        } b;
152
} atapi_feature_reg_t;
153
 
154
/*
155
 *      ATAPI Byte Count Register.
156
 */
157
typedef union {
158
        u16 all                 : 16;
159
        struct {
160
#if defined(__LITTLE_ENDIAN_BITFIELD)
161
                u8 low;                 /* LSB */
162
                u8 high;                /* MSB */
163
#elif defined(__BIG_ENDIAN_BITFIELD)
164
                u8 high;
165
                u8 low;
166
#else
167
#error "Please fix <asm/byteorder.h>"
168
#endif
169
        } b;
170
} atapi_bcount_reg_t;
171
 
172
/*
173
 *      ATAPI Interrupt Reason Register.
174
 */
175
typedef union {
176
        u8 all                  : 8;
177
        struct {
178
#if defined(__LITTLE_ENDIAN_BITFIELD)
179
                u8 cod          : 1;    /* Information transferred is command (1) or data (0) */
180
                u8 io           : 1;    /* The device requests us to read (1) or write (0) */
181
                u8 reserved     : 6;    /* Reserved */
182
#elif defined(__BIG_ENDIAN_BITFIELD)
183
                u8 reserved     : 6;
184
                u8 io           : 1;
185
                u8 cod          : 1;
186
#else
187
#error "Please fix <asm/byteorder.h>"
188
#endif
189
        } b;
190
} atapi_ireason_reg_t;
191
 
192
/* Currently unused, but please do not remove.  --bkz */
193
/*
194
 *      ATAPI Drive Select Register.
195
 */
196
typedef union {
197
        u8 all                  :8;
198
        struct {
199
#if defined(__LITTLE_ENDIAN_BITFIELD)
200
                u8 sam_lun      :3;     /* Logical unit number */
201
                u8 reserved3    :1;     /* Reserved */
202
                u8 drv          :1;     /* The responding drive will be drive 0 (0) or drive 1 (1) */
203
                u8 one5         :1;     /* Should be set to 1 */
204
                u8 reserved6    :1;     /* Reserved */
205
                u8 one7         :1;     /* Should be set to 1 */
206
#elif defined(__BIG_ENDIAN_BITFIELD)
207
                u8 one7         :1;
208
                u8 reserved6    :1;
209
                u8 one5         :1;
210
                u8 drv          :1;
211
                u8 reserved3    :1;
212
                u8 sam_lun      :3;
213
#else
214
#error "Please fix <asm/byteorder.h>"
215
#endif
216
        } b;
217
} atapi_drivesel_reg_t;
218
 
219
/* Currently unused, but please do not remove.  --bkz */
220
/*
221
 *      ATAPI Device Control Register.
222
 */
223
typedef union {
224
        u8 all                  : 8;
225
        struct {
226
#if defined(__LITTLE_ENDIAN_BITFIELD)
227
                u8 zero0        : 1;    /* Should be set to zero */
228
                u8 nien         : 1;    /* Device interrupt is disabled (1) or enabled (0) */
229
                u8 srst         : 1;    /* ATA software reset. ATAPI devices should use the new ATAPI srst. */
230
                u8 one3         : 1;    /* Should be set to 1 */
231
                u8 reserved4567 : 4;    /* Reserved */
232
#elif defined(__BIG_ENDIAN_BITFIELD)
233
                u8 reserved4567 : 4;
234
                u8 one3         : 1;
235
                u8 srst         : 1;
236
                u8 nien         : 1;
237
                u8 zero0        : 1;
238
#else
239
#error "Please fix <asm/byteorder.h>"
240
#endif
241
        } b;
242
} atapi_control_reg_t;
243
 
244
/*
245
 *      The following is used to format the general configuration word
246
 *      of the ATAPI IDENTIFY DEVICE command.
247
 */
248
struct atapi_id_gcw {
249
#if defined(__LITTLE_ENDIAN_BITFIELD)
250
        u8 packet_size          : 2;    /* Packet Size */
251
        u8 reserved234          : 3;    /* Reserved */
252
        u8 drq_type             : 2;    /* Command packet DRQ type */
253
        u8 removable            : 1;    /* Removable media */
254
        u8 device_type          : 5;    /* Device type */
255
        u8 reserved13           : 1;    /* Reserved */
256
        u8 protocol             : 2;    /* Protocol type */
257
#elif defined(__BIG_ENDIAN_BITFIELD)
258
        u8 protocol             : 2;
259
        u8 reserved13           : 1;
260
        u8 device_type          : 5;
261
        u8 removable            : 1;
262
        u8 drq_type             : 2;
263
        u8 reserved234          : 3;
264
        u8 packet_size          : 2;
265
#else
266
#error "Please fix <asm/byteorder.h>"
267
#endif
268
};
269
 
270
/*
271
 *      INQUIRY packet command - Data Format.
272
 */
273
typedef struct {
274
#if defined(__LITTLE_ENDIAN_BITFIELD)
275
        u8      device_type     : 5;    /* Peripheral Device Type */
276
        u8      reserved0_765   : 3;    /* Peripheral Qualifier - Reserved */
277
        u8      reserved1_6t0   : 7;    /* Reserved */
278
        u8      rmb             : 1;    /* Removable Medium Bit */
279
        u8      ansi_version    : 3;    /* ANSI Version */
280
        u8      ecma_version    : 3;    /* ECMA Version */
281
        u8      iso_version     : 2;    /* ISO Version */
282
        u8      response_format : 4;    /* Response Data Format */
283
        u8      reserved3_45    : 2;    /* Reserved */
284
        u8      reserved3_6     : 1;    /* TrmIOP - Reserved */
285
        u8      reserved3_7     : 1;    /* AENC - Reserved */
286
#elif defined(__BIG_ENDIAN_BITFIELD)
287
        u8      reserved0_765   : 3;
288
        u8      device_type     : 5;
289
        u8      rmb             : 1;
290
        u8      reserved1_6t0   : 7;
291
        u8      iso_version     : 2;
292
        u8      ecma_version    : 3;
293
        u8      ansi_version    : 3;
294
        u8      reserved3_7     : 1;
295
        u8      reserved3_6     : 1;
296
        u8      reserved3_45    : 2;
297
        u8      response_format : 4;
298
#else
299
#error "Please fix <asm/byteorder.h>"
300
#endif
301
        u8      additional_length;      /* Additional Length (total_length-4) */
302
        u8      rsv5, rsv6, rsv7;       /* Reserved */
303
        u8      vendor_id[8];           /* Vendor Identification */
304
        u8      product_id[16];         /* Product Identification */
305
        u8      revision_level[4];      /* Revision Level */
306
        u8      vendor_specific[20];    /* Vendor Specific - Optional */
307
        u8      reserved56t95[40];      /* Reserved - Optional */
308
                                        /* Additional information may be returned */
309
} atapi_inquiry_result_t;
310
 
311
/*
312
 *      REQUEST SENSE packet command result - Data Format.
313
 */
314
typedef struct atapi_request_sense {
315
#if defined(__LITTLE_ENDIAN_BITFIELD)
316
        u8      error_code      : 7;    /* Error Code (0x70 - current or 0x71 - deferred) */
317
        u8      valid           : 1;    /* The information field conforms to standard */
318
        u8      reserved1       : 8;    /* Reserved (Segment Number) */
319
        u8      sense_key       : 4;    /* Sense Key */
320
        u8      reserved2_4     : 1;    /* Reserved */
321
        u8      ili             : 1;    /* Incorrect Length Indicator */
322
        u8      eom             : 1;    /* End Of Medium */
323
        u8      filemark        : 1;    /* Filemark */
324
#elif defined(__BIG_ENDIAN_BITFIELD)
325
        u8      valid           : 1;
326
        u8      error_code      : 7;
327
        u8      reserved1       : 8;
328
        u8      filemark        : 1;
329
        u8      eom             : 1;
330
        u8      ili             : 1;
331
        u8      reserved2_4     : 1;
332
        u8      sense_key       : 4;
333
#else
334
#error "Please fix <asm/byteorder.h>"
335
#endif
336
        u32     information __attribute__ ((packed));
337
        u8      asl;                    /* Additional sense length (n-7) */
338
        u32     command_specific;       /* Additional command specific information */
339
        u8      asc;                    /* Additional Sense Code */
340
        u8      ascq;                   /* Additional Sense Code Qualifier */
341
        u8      replaceable_unit_code;  /* Field Replaceable Unit Code */
342
#if defined(__LITTLE_ENDIAN_BITFIELD)
343
        u8      sk_specific1    : 7;    /* Sense Key Specific */
344
        u8      sksv            : 1;    /* Sense Key Specific information is valid */
345
#elif defined(__BIG_ENDIAN_BITFIELD)
346
        u8      sksv            : 1;    /* Sense Key Specific information is valid */
347
        u8      sk_specific1    : 7;    /* Sense Key Specific */
348
#else
349
#error "Please fix <asm/byteorder.h>"
350
#endif
351
        u8      sk_specific[2];         /* Sense Key Specific */
352
        u8      pad[2];                 /* Padding to 20 bytes */
353
} atapi_request_sense_result_t;
354
 
355
 
356
extern void atapi_init_pc(struct atapi_packet_command *pc);
357
 
358
extern void atapi_discard_data(struct ata_device *, unsigned int);
359
extern void atapi_write_zeros(struct ata_device *, unsigned int);
360
 
361
extern void atapi_read(struct ata_device *, u8 *, unsigned int);
362
extern void atapi_write(struct ata_device *, u8 *, unsigned int);
363
 
364
typedef enum {
365
        ide_wait,       /* insert rq at end of list, and wait for it */
366
        ide_preempt,    /* insert rq in front of current request */
367
        ide_end         /* insert rq at end of list, but don't wait for it */
368
} ide_action_t;
369
 
370
extern int ide_do_drive_cmd(struct ata_device *, struct request *, ide_action_t);