sd.c revision 1.1 1 1.1 kiyohara /* $NetBSD: sd.c,v 1.1 2010/10/14 06:58:22 kiyohara Exp $ */
2 1.1 kiyohara /*
3 1.1 kiyohara * Copyright (c) 2010 KIYOHARA Takashi
4 1.1 kiyohara * All rights reserved.
5 1.1 kiyohara *
6 1.1 kiyohara * Redistribution and use in source and binary forms, with or without
7 1.1 kiyohara * modification, are permitted provided that the following conditions
8 1.1 kiyohara * are met:
9 1.1 kiyohara * 1. Redistributions of source code must retain the above copyright
10 1.1 kiyohara * notice, this list of conditions and the following disclaimer.
11 1.1 kiyohara * 2. Redistributions in binary form must reproduce the above copyright
12 1.1 kiyohara * notice, this list of conditions and the following disclaimer in the
13 1.1 kiyohara * documentation and/or other materials provided with the distribution.
14 1.1 kiyohara *
15 1.1 kiyohara * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
16 1.1 kiyohara * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
17 1.1 kiyohara * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
18 1.1 kiyohara * DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT,
19 1.1 kiyohara * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
20 1.1 kiyohara * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
21 1.1 kiyohara * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
22 1.1 kiyohara * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
23 1.1 kiyohara * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
24 1.1 kiyohara * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
25 1.1 kiyohara * POSSIBILITY OF SUCH DAMAGE.
26 1.1 kiyohara */
27 1.1 kiyohara
28 1.1 kiyohara #include <sys/param.h>
29 1.1 kiyohara #include <sys/types.h>
30 1.1 kiyohara #include <sys/stdint.h>
31 1.1 kiyohara
32 1.1 kiyohara #include <lib/libsa/stand.h>
33 1.1 kiyohara #include <lib/libkern/libkern.h>
34 1.1 kiyohara
35 1.1 kiyohara #include <machine/param.h>
36 1.1 kiyohara #include <machine/stdarg.h>
37 1.1 kiyohara
38 1.1 kiyohara #include "boot.h"
39 1.1 kiyohara #include "sdvar.h"
40 1.1 kiyohara
41 1.1 kiyohara #ifdef DEBUG
42 1.1 kiyohara #define DPRINTF(x) printf x
43 1.1 kiyohara #else
44 1.1 kiyohara #define DPRINTF(x)
45 1.1 kiyohara #endif
46 1.1 kiyohara
47 1.1 kiyohara #define SD_DEFAULT_BLKSIZE 512
48 1.1 kiyohara
49 1.1 kiyohara
50 1.1 kiyohara struct sd_mode_sense_data {
51 1.1 kiyohara struct scsi_mode_parameter_header_6 header;
52 1.1 kiyohara struct scsi_general_block_descriptor blk_desc;
53 1.1 kiyohara union scsi_disk_pages pages;
54 1.1 kiyohara };
55 1.1 kiyohara
56 1.1 kiyohara static int sd_validate_blksize(int);
57 1.1 kiyohara static uint64_t sd_read_capacity(struct sd_softc *, int *);
58 1.1 kiyohara static int sd_get_simplifiedparms(struct sd_softc *);
59 1.1 kiyohara static int sd_get_capacity(struct sd_softc *);
60 1.1 kiyohara static int sd_get_parms_page4(struct sd_softc *, struct disk_parms *);
61 1.1 kiyohara static int sd_get_parms_page5(struct sd_softc *, struct disk_parms *);
62 1.1 kiyohara static int sd_get_parms(struct sd_softc *);
63 1.1 kiyohara static void sdgetdefaultlabel(struct sd_softc *, struct disklabel *);
64 1.1 kiyohara static int sdgetdisklabel(struct sd_softc *);
65 1.1 kiyohara
66 1.1 kiyohara int sdopen(struct open_file *, ...);
67 1.1 kiyohara int sdclose(struct open_file *);
68 1.1 kiyohara int sdstrategy(void *, int, daddr_t, size_t, void *, size_t *);
69 1.1 kiyohara
70 1.1 kiyohara
71 1.1 kiyohara static int
72 1.1 kiyohara sd_validate_blksize(int len)
73 1.1 kiyohara {
74 1.1 kiyohara
75 1.1 kiyohara switch (len) {
76 1.1 kiyohara case 256:
77 1.1 kiyohara case 512:
78 1.1 kiyohara case 1024:
79 1.1 kiyohara case 2048:
80 1.1 kiyohara case 4096:
81 1.1 kiyohara return 1;
82 1.1 kiyohara }
83 1.1 kiyohara return 0;
84 1.1 kiyohara }
85 1.1 kiyohara
86 1.1 kiyohara /*
87 1.1 kiyohara * sd_read_capacity:
88 1.1 kiyohara *
89 1.1 kiyohara * Find out from the device what its capacity is.
90 1.1 kiyohara */
91 1.1 kiyohara static uint64_t
92 1.1 kiyohara sd_read_capacity(struct sd_softc *sd, int *blksize)
93 1.1 kiyohara {
94 1.1 kiyohara union {
95 1.1 kiyohara struct scsipi_read_capacity_10 cmd;
96 1.1 kiyohara struct scsipi_read_capacity_16 cmd16;
97 1.1 kiyohara } cmd;
98 1.1 kiyohara union {
99 1.1 kiyohara struct scsipi_read_capacity_10_data data;
100 1.1 kiyohara struct scsipi_read_capacity_16_data data16;
101 1.1 kiyohara } data;
102 1.1 kiyohara uint64_t rv;
103 1.1 kiyohara
104 1.1 kiyohara memset(&cmd, 0, sizeof(cmd));
105 1.1 kiyohara cmd.cmd.opcode = READ_CAPACITY_10;
106 1.1 kiyohara
107 1.1 kiyohara /*
108 1.1 kiyohara * If the command works, interpret the result as a 4 byte
109 1.1 kiyohara * number of blocks
110 1.1 kiyohara */
111 1.1 kiyohara rv = 0;
112 1.1 kiyohara memset(&data, 0, sizeof(data.data));
113 1.1 kiyohara if (scsi_command(sd, (void *)&cmd.cmd, sizeof(cmd.cmd),
114 1.1 kiyohara (void *)&data, sizeof(data.data)) != 0)
115 1.1 kiyohara goto out;
116 1.1 kiyohara
117 1.1 kiyohara if (_4btol(data.data.addr) != 0xffffffff) {
118 1.1 kiyohara *blksize = _4btol(data.data.length);
119 1.1 kiyohara rv = _4btol(data.data.addr) + 1;
120 1.1 kiyohara goto out;
121 1.1 kiyohara }
122 1.1 kiyohara
123 1.1 kiyohara /*
124 1.1 kiyohara * Device is larger than can be reflected by READ CAPACITY (10).
125 1.1 kiyohara * Try READ CAPACITY (16).
126 1.1 kiyohara */
127 1.1 kiyohara
128 1.1 kiyohara memset(&cmd, 0, sizeof(cmd));
129 1.1 kiyohara cmd.cmd16.opcode = READ_CAPACITY_16;
130 1.1 kiyohara cmd.cmd16.byte2 = SRC16_SERVICE_ACTION;
131 1.1 kiyohara _lto4b(sizeof(data.data16), cmd.cmd16.len);
132 1.1 kiyohara
133 1.1 kiyohara memset(&data, 0, sizeof(data.data16));
134 1.1 kiyohara if (scsi_command(sd, (void *)&cmd.cmd16, sizeof(cmd.cmd16),
135 1.1 kiyohara (void *)&data, sizeof(data.data16)) != 0)
136 1.1 kiyohara goto out;
137 1.1 kiyohara
138 1.1 kiyohara *blksize = _4btol(data.data16.length);
139 1.1 kiyohara rv = _8btol(data.data16.addr) + 1;
140 1.1 kiyohara
141 1.1 kiyohara out:
142 1.1 kiyohara return rv;
143 1.1 kiyohara }
144 1.1 kiyohara
145 1.1 kiyohara static int
146 1.1 kiyohara sd_get_simplifiedparms(struct sd_softc *sd)
147 1.1 kiyohara {
148 1.1 kiyohara struct {
149 1.1 kiyohara struct scsi_mode_parameter_header_6 header;
150 1.1 kiyohara /* no block descriptor */
151 1.1 kiyohara uint8_t pg_code; /* page code (should be 6) */
152 1.1 kiyohara uint8_t pg_length; /* page length (should be 11) */
153 1.1 kiyohara uint8_t wcd; /* bit0: cache disable */
154 1.1 kiyohara uint8_t lbs[2]; /* logical block size */
155 1.1 kiyohara uint8_t size[5]; /* number of log. blocks */
156 1.1 kiyohara uint8_t pp; /* power/performance */
157 1.1 kiyohara uint8_t flags;
158 1.1 kiyohara uint8_t resvd;
159 1.1 kiyohara } scsipi_sense;
160 1.1 kiyohara struct disk_parms *dp = &sd->sc_params;
161 1.1 kiyohara uint64_t blocks;
162 1.1 kiyohara int error, blksize;
163 1.1 kiyohara
164 1.1 kiyohara /*
165 1.1 kiyohara * sd_read_capacity (ie "read capacity") and mode sense page 6
166 1.1 kiyohara * give the same information. Do both for now, and check
167 1.1 kiyohara * for consistency.
168 1.1 kiyohara * XXX probably differs for removable media
169 1.1 kiyohara */
170 1.1 kiyohara dp->blksize = SD_DEFAULT_BLKSIZE;
171 1.1 kiyohara if ((blocks = sd_read_capacity(sd, &blksize)) == 0)
172 1.1 kiyohara return SDGP_RESULT_OFFLINE; /* XXX? */
173 1.1 kiyohara
174 1.1 kiyohara error = scsi_mode_sense(sd, SMS_DBD, 6,
175 1.1 kiyohara &scsipi_sense.header, sizeof(scsipi_sense));
176 1.1 kiyohara
177 1.1 kiyohara if (error != 0)
178 1.1 kiyohara return SDGP_RESULT_OFFLINE; /* XXX? */
179 1.1 kiyohara
180 1.1 kiyohara dp->blksize = blksize;
181 1.1 kiyohara if (!sd_validate_blksize(dp->blksize))
182 1.1 kiyohara dp->blksize = _2btol(scsipi_sense.lbs);
183 1.1 kiyohara if (!sd_validate_blksize(dp->blksize))
184 1.1 kiyohara dp->blksize = SD_DEFAULT_BLKSIZE;
185 1.1 kiyohara
186 1.1 kiyohara /*
187 1.1 kiyohara * Create a pseudo-geometry.
188 1.1 kiyohara */
189 1.1 kiyohara dp->heads = 64;
190 1.1 kiyohara dp->sectors = 32;
191 1.1 kiyohara dp->cyls = blocks / (dp->heads * dp->sectors);
192 1.1 kiyohara dp->disksize = _5btol(scsipi_sense.size);
193 1.1 kiyohara if (dp->disksize <= UINT32_MAX && dp->disksize != blocks) {
194 1.1 kiyohara printf("RBC size: mode sense=%llu, get cap=%llu\n",
195 1.1 kiyohara (unsigned long long)dp->disksize,
196 1.1 kiyohara (unsigned long long)blocks);
197 1.1 kiyohara dp->disksize = blocks;
198 1.1 kiyohara }
199 1.1 kiyohara dp->disksize512 = (dp->disksize * dp->blksize) / DEV_BSIZE;
200 1.1 kiyohara
201 1.1 kiyohara return SDGP_RESULT_OK;
202 1.1 kiyohara }
203 1.1 kiyohara
204 1.1 kiyohara /*
205 1.1 kiyohara * Get the scsi driver to send a full inquiry to the * device and use the
206 1.1 kiyohara * results to fill out the disk parameter structure.
207 1.1 kiyohara */
208 1.1 kiyohara static int
209 1.1 kiyohara sd_get_capacity(struct sd_softc *sd)
210 1.1 kiyohara {
211 1.1 kiyohara struct disk_parms *dp = &sd->sc_params;
212 1.1 kiyohara uint64_t blocks;
213 1.1 kiyohara int error, blksize;
214 1.1 kiyohara
215 1.1 kiyohara dp->disksize = blocks = sd_read_capacity(sd, &blksize);
216 1.1 kiyohara if (blocks == 0) {
217 1.1 kiyohara struct scsipi_read_format_capacities cmd;
218 1.1 kiyohara struct {
219 1.1 kiyohara struct scsipi_capacity_list_header header;
220 1.1 kiyohara struct scsipi_capacity_descriptor desc;
221 1.1 kiyohara } __packed data;
222 1.1 kiyohara
223 1.1 kiyohara memset(&cmd, 0, sizeof(cmd));
224 1.1 kiyohara memset(&data, 0, sizeof(data));
225 1.1 kiyohara cmd.opcode = READ_FORMAT_CAPACITIES;
226 1.1 kiyohara _lto2b(sizeof(data), cmd.length);
227 1.1 kiyohara
228 1.1 kiyohara error = scsi_command(sd,
229 1.1 kiyohara (void *)&cmd, sizeof(cmd), (void *)&data, sizeof(data));
230 1.1 kiyohara if (error == EFTYPE)
231 1.1 kiyohara /* Medium Format Corrupted, handle as not formatted */
232 1.1 kiyohara return SDGP_RESULT_UNFORMATTED;
233 1.1 kiyohara if (error || data.header.length == 0)
234 1.1 kiyohara return SDGP_RESULT_OFFLINE;
235 1.1 kiyohara
236 1.1 kiyohara switch (data.desc.byte5 & SCSIPI_CAP_DESC_CODE_MASK) {
237 1.1 kiyohara case SCSIPI_CAP_DESC_CODE_RESERVED:
238 1.1 kiyohara case SCSIPI_CAP_DESC_CODE_FORMATTED:
239 1.1 kiyohara break;
240 1.1 kiyohara
241 1.1 kiyohara case SCSIPI_CAP_DESC_CODE_UNFORMATTED:
242 1.1 kiyohara return SDGP_RESULT_UNFORMATTED;
243 1.1 kiyohara
244 1.1 kiyohara case SCSIPI_CAP_DESC_CODE_NONE:
245 1.1 kiyohara return SDGP_RESULT_OFFLINE;
246 1.1 kiyohara }
247 1.1 kiyohara
248 1.1 kiyohara dp->disksize = blocks = _4btol(data.desc.nblks);
249 1.1 kiyohara if (blocks == 0)
250 1.1 kiyohara return SDGP_RESULT_OFFLINE; /* XXX? */
251 1.1 kiyohara
252 1.1 kiyohara blksize = _3btol(data.desc.blklen);
253 1.1 kiyohara
254 1.1 kiyohara } else if (!sd_validate_blksize(blksize)) {
255 1.1 kiyohara struct sd_mode_sense_data scsipi_sense;
256 1.1 kiyohara int bsize;
257 1.1 kiyohara
258 1.1 kiyohara memset(&scsipi_sense, 0, sizeof(scsipi_sense));
259 1.1 kiyohara error = scsi_mode_sense(sd, 0, 0, &scsipi_sense.header,
260 1.1 kiyohara sizeof(struct scsi_mode_parameter_header_6) +
261 1.1 kiyohara sizeof(scsipi_sense.blk_desc));
262 1.1 kiyohara if (!error) {
263 1.1 kiyohara bsize = scsipi_sense.header.blk_desc_len;
264 1.1 kiyohara
265 1.1 kiyohara if (bsize >= 8)
266 1.1 kiyohara blksize = _3btol(scsipi_sense.blk_desc.blklen);
267 1.1 kiyohara }
268 1.1 kiyohara }
269 1.1 kiyohara
270 1.1 kiyohara if (!sd_validate_blksize(blksize))
271 1.1 kiyohara blksize = SD_DEFAULT_BLKSIZE;
272 1.1 kiyohara
273 1.1 kiyohara dp->blksize = blksize;
274 1.1 kiyohara dp->disksize512 = (blocks * dp->blksize) / DEV_BSIZE;
275 1.1 kiyohara return 0;
276 1.1 kiyohara }
277 1.1 kiyohara
278 1.1 kiyohara static int
279 1.1 kiyohara sd_get_parms_page4(struct sd_softc *sd, struct disk_parms *dp)
280 1.1 kiyohara {
281 1.1 kiyohara struct sd_mode_sense_data scsipi_sense;
282 1.1 kiyohara union scsi_disk_pages *pages;
283 1.1 kiyohara size_t poffset;
284 1.1 kiyohara int byte2, error;
285 1.1 kiyohara
286 1.1 kiyohara byte2 = SMS_DBD;
287 1.1 kiyohara again:
288 1.1 kiyohara memset(&scsipi_sense, 0, sizeof(scsipi_sense));
289 1.1 kiyohara error = scsi_mode_sense(sd, byte2, 4, &scsipi_sense.header,
290 1.1 kiyohara (byte2 ? 0 : sizeof(scsipi_sense.blk_desc)) +
291 1.1 kiyohara sizeof(scsipi_sense.pages.rigid_geometry));
292 1.1 kiyohara if (error) {
293 1.1 kiyohara if (byte2 == SMS_DBD) {
294 1.1 kiyohara /* No result; try once more with DBD off */
295 1.1 kiyohara byte2 = 0;
296 1.1 kiyohara goto again;
297 1.1 kiyohara }
298 1.1 kiyohara return error;
299 1.1 kiyohara }
300 1.1 kiyohara
301 1.1 kiyohara poffset = sizeof(scsipi_sense.header);
302 1.1 kiyohara poffset += scsipi_sense.header.blk_desc_len;
303 1.1 kiyohara
304 1.1 kiyohara if (poffset > sizeof(scsipi_sense) - sizeof(pages->rigid_geometry))
305 1.1 kiyohara return ERESTART;
306 1.1 kiyohara
307 1.1 kiyohara pages = (void *)((u_long)&scsipi_sense + poffset);
308 1.1 kiyohara #if 0
309 1.1 kiyohara {
310 1.1 kiyohara size_t i;
311 1.1 kiyohara u_int8_t *p;
312 1.1 kiyohara
313 1.1 kiyohara printf("page 4 sense:");
314 1.1 kiyohara for (i = sizeof(scsipi_sense), p = (void *)&scsipi_sense; i;
315 1.1 kiyohara i--, p++)
316 1.1 kiyohara printf(" %02x", *p);
317 1.1 kiyohara printf("\n");
318 1.1 kiyohara printf("page 4 pg_code=%d sense=%p/%p\n",
319 1.1 kiyohara pages->rigid_geometry.pg_code, &scsipi_sense, pages);
320 1.1 kiyohara }
321 1.1 kiyohara #endif
322 1.1 kiyohara
323 1.1 kiyohara if ((pages->rigid_geometry.pg_code & PGCODE_MASK) != 4)
324 1.1 kiyohara return ERESTART;
325 1.1 kiyohara
326 1.1 kiyohara /*
327 1.1 kiyohara * KLUDGE!! (for zone recorded disks)
328 1.1 kiyohara * give a number of sectors so that sec * trks * cyls
329 1.1 kiyohara * is <= disk_size
330 1.1 kiyohara * can lead to wasted space! THINK ABOUT THIS !
331 1.1 kiyohara */
332 1.1 kiyohara dp->heads = pages->rigid_geometry.nheads;
333 1.1 kiyohara dp->cyls = _3btol(pages->rigid_geometry.ncyl);
334 1.1 kiyohara if (dp->heads == 0 || dp->cyls == 0)
335 1.1 kiyohara return ERESTART;
336 1.1 kiyohara dp->sectors = dp->disksize / (dp->heads * dp->cyls); /* XXX */
337 1.1 kiyohara
338 1.1 kiyohara dp->rot_rate = _2btol(pages->rigid_geometry.rpm);
339 1.1 kiyohara if (dp->rot_rate == 0)
340 1.1 kiyohara dp->rot_rate = 3600;
341 1.1 kiyohara
342 1.1 kiyohara #if 0
343 1.1 kiyohara printf("page 4 ok\n");
344 1.1 kiyohara #endif
345 1.1 kiyohara return 0;
346 1.1 kiyohara }
347 1.1 kiyohara
348 1.1 kiyohara static int
349 1.1 kiyohara sd_get_parms_page5(struct sd_softc *sd, struct disk_parms *dp)
350 1.1 kiyohara {
351 1.1 kiyohara struct sd_mode_sense_data scsipi_sense;
352 1.1 kiyohara union scsi_disk_pages *pages;
353 1.1 kiyohara size_t poffset;
354 1.1 kiyohara int byte2, error;
355 1.1 kiyohara
356 1.1 kiyohara byte2 = SMS_DBD;
357 1.1 kiyohara again:
358 1.1 kiyohara memset(&scsipi_sense, 0, sizeof(scsipi_sense));
359 1.1 kiyohara error = scsi_mode_sense(sd, 0, 5, &scsipi_sense.header,
360 1.1 kiyohara (byte2 ? 0 : sizeof(scsipi_sense.blk_desc)) +
361 1.1 kiyohara sizeof(scsipi_sense.pages.flex_geometry));
362 1.1 kiyohara if (error) {
363 1.1 kiyohara if (byte2 == SMS_DBD) {
364 1.1 kiyohara /* No result; try once more with DBD off */
365 1.1 kiyohara byte2 = 0;
366 1.1 kiyohara goto again;
367 1.1 kiyohara }
368 1.1 kiyohara return error;
369 1.1 kiyohara }
370 1.1 kiyohara
371 1.1 kiyohara poffset = sizeof(scsipi_sense.header);
372 1.1 kiyohara poffset += scsipi_sense.header.blk_desc_len;
373 1.1 kiyohara
374 1.1 kiyohara if (poffset > sizeof(scsipi_sense) - sizeof(pages->flex_geometry))
375 1.1 kiyohara return ERESTART;
376 1.1 kiyohara
377 1.1 kiyohara pages = (void *)((u_long)&scsipi_sense + poffset);
378 1.1 kiyohara #if 0
379 1.1 kiyohara {
380 1.1 kiyohara size_t i;
381 1.1 kiyohara u_int8_t *p;
382 1.1 kiyohara
383 1.1 kiyohara printf("page 5 sense:");
384 1.1 kiyohara for (i = sizeof(scsipi_sense), p = (void *)&scsipi_sense; i;
385 1.1 kiyohara i--, p++)
386 1.1 kiyohara printf(" %02x", *p);
387 1.1 kiyohara printf("\n");
388 1.1 kiyohara printf("page 5 pg_code=%d sense=%p/%p\n",
389 1.1 kiyohara pages->flex_geometry.pg_code, &scsipi_sense, pages);
390 1.1 kiyohara }
391 1.1 kiyohara #endif
392 1.1 kiyohara
393 1.1 kiyohara if ((pages->flex_geometry.pg_code & PGCODE_MASK) != 5)
394 1.1 kiyohara return ERESTART;
395 1.1 kiyohara
396 1.1 kiyohara dp->heads = pages->flex_geometry.nheads;
397 1.1 kiyohara dp->cyls = _2btol(pages->flex_geometry.ncyl);
398 1.1 kiyohara dp->sectors = pages->flex_geometry.ph_sec_tr;
399 1.1 kiyohara if (dp->heads == 0 || dp->cyls == 0 || dp->sectors == 0)
400 1.1 kiyohara return ERESTART;
401 1.1 kiyohara
402 1.1 kiyohara dp->rot_rate = _2btol(pages->rigid_geometry.rpm);
403 1.1 kiyohara if (dp->rot_rate == 0)
404 1.1 kiyohara dp->rot_rate = 3600;
405 1.1 kiyohara
406 1.1 kiyohara #if 0
407 1.1 kiyohara printf("page 5 ok\n");
408 1.1 kiyohara #endif
409 1.1 kiyohara return 0;
410 1.1 kiyohara }
411 1.1 kiyohara
412 1.1 kiyohara static int
413 1.1 kiyohara sd_get_parms(struct sd_softc *sd)
414 1.1 kiyohara {
415 1.1 kiyohara struct disk_parms *dp = &sd->sc_params;
416 1.1 kiyohara int error;
417 1.1 kiyohara
418 1.1 kiyohara /*
419 1.1 kiyohara * If offline, the SDEV_MEDIA_LOADED flag will be
420 1.1 kiyohara * cleared by the caller if necessary.
421 1.1 kiyohara */
422 1.1 kiyohara if (sd->sc_type == T_SIMPLE_DIRECT) {
423 1.1 kiyohara error = sd_get_simplifiedparms(sd);
424 1.1 kiyohara if (error)
425 1.1 kiyohara return error;
426 1.1 kiyohara goto ok;
427 1.1 kiyohara }
428 1.1 kiyohara
429 1.1 kiyohara error = sd_get_capacity(sd);
430 1.1 kiyohara if (error)
431 1.1 kiyohara return error;
432 1.1 kiyohara
433 1.1 kiyohara if (sd->sc_type == T_OPTICAL)
434 1.1 kiyohara goto page0;
435 1.1 kiyohara
436 1.1 kiyohara if (sd->sc_flags & FLAGS_REMOVABLE) {
437 1.1 kiyohara if (!sd_get_parms_page5(sd, dp) ||
438 1.1 kiyohara !sd_get_parms_page4(sd, dp))
439 1.1 kiyohara goto ok;
440 1.1 kiyohara } else {
441 1.1 kiyohara if (!sd_get_parms_page4(sd, dp) ||
442 1.1 kiyohara !sd_get_parms_page5(sd, dp))
443 1.1 kiyohara goto ok;
444 1.1 kiyohara }
445 1.1 kiyohara
446 1.1 kiyohara page0:
447 1.1 kiyohara printf("fabricating a geometry\n");
448 1.1 kiyohara /* Try calling driver's method for figuring out geometry. */
449 1.1 kiyohara /*
450 1.1 kiyohara * Use adaptec standard fictitious geometry
451 1.1 kiyohara * this depends on which controller (e.g. 1542C is
452 1.1 kiyohara * different. but we have to put SOMETHING here..)
453 1.1 kiyohara */
454 1.1 kiyohara dp->heads = 64;
455 1.1 kiyohara dp->sectors = 32;
456 1.1 kiyohara dp->cyls = dp->disksize / (64 * 32);
457 1.1 kiyohara dp->rot_rate = 3600;
458 1.1 kiyohara
459 1.1 kiyohara ok:
460 1.1 kiyohara DPRINTF(("disksize = %" PRId64 ", disksize512 = %" PRId64 ".\n",
461 1.1 kiyohara dp->disksize, dp->disksize512));
462 1.1 kiyohara
463 1.1 kiyohara return 0;
464 1.1 kiyohara }
465 1.1 kiyohara
466 1.1 kiyohara static void
467 1.1 kiyohara sdgetdefaultlabel(struct sd_softc *sd, struct disklabel *lp)
468 1.1 kiyohara {
469 1.1 kiyohara
470 1.1 kiyohara memset(lp, 0, sizeof(struct disklabel));
471 1.1 kiyohara
472 1.1 kiyohara lp->d_secsize = sd->sc_params.blksize;
473 1.1 kiyohara lp->d_ntracks = sd->sc_params.heads;
474 1.1 kiyohara lp->d_nsectors = sd->sc_params.sectors;
475 1.1 kiyohara lp->d_ncylinders = sd->sc_params.cyls;
476 1.1 kiyohara lp->d_secpercyl = lp->d_ntracks * lp->d_nsectors;
477 1.1 kiyohara
478 1.1 kiyohara lp->d_type = DTYPE_SCSI;
479 1.1 kiyohara
480 1.1 kiyohara strncpy(lp->d_packname, "fictitious", 16);
481 1.1 kiyohara lp->d_secperunit = sd->sc_params.disksize;
482 1.1 kiyohara lp->d_rpm = sd->sc_params.rot_rate;
483 1.1 kiyohara lp->d_interleave = 1;
484 1.1 kiyohara lp->d_flags = (sd->sc_flags & FLAGS_REMOVABLE) ? D_REMOVABLE : 0;
485 1.1 kiyohara
486 1.1 kiyohara lp->d_partitions[RAW_PART].p_offset = 0;
487 1.1 kiyohara lp->d_partitions[RAW_PART].p_size = lp->d_secperunit;
488 1.1 kiyohara lp->d_partitions[RAW_PART].p_fstype = FS_UNUSED;
489 1.1 kiyohara lp->d_npartitions = RAW_PART + 1;
490 1.1 kiyohara
491 1.1 kiyohara lp->d_magic = DISKMAGIC;
492 1.1 kiyohara lp->d_magic2 = DISKMAGIC;
493 1.1 kiyohara lp->d_checksum = dkcksum(lp);
494 1.1 kiyohara }
495 1.1 kiyohara
496 1.1 kiyohara /*
497 1.1 kiyohara * Load the label information on the named device.
498 1.1 kiyohara */
499 1.1 kiyohara static int
500 1.1 kiyohara sdgetdisklabel(struct sd_softc *sd)
501 1.1 kiyohara {
502 1.1 kiyohara struct mbr_sector *mbr;
503 1.1 kiyohara struct mbr_partition *mp;
504 1.1 kiyohara struct disklabel *lp = &sd->sc_label;
505 1.1 kiyohara size_t rsize;
506 1.1 kiyohara int sector, i;
507 1.1 kiyohara char *msg;
508 1.1 kiyohara uint8_t buf[DEV_BSIZE];
509 1.1 kiyohara
510 1.1 kiyohara sdgetdefaultlabel(sd, lp);
511 1.1 kiyohara
512 1.1 kiyohara if (lp->d_secpercyl == 0) {
513 1.1 kiyohara lp->d_secpercyl = 100;
514 1.1 kiyohara /* as long as it's not 0 - readdisklabel divides by it (?) */
515 1.1 kiyohara }
516 1.1 kiyohara
517 1.1 kiyohara /*
518 1.1 kiyohara * Find NetBSD Partition in DOS partition table.
519 1.1 kiyohara */
520 1.1 kiyohara sector = 0;
521 1.1 kiyohara if (sdstrategy(sd, F_READ, MBR_BBSECTOR, DEV_BSIZE, buf, &rsize))
522 1.1 kiyohara return EOFFSET;
523 1.1 kiyohara
524 1.1 kiyohara mbr = (struct mbr_sector *)buf;
525 1.1 kiyohara if (mbr->mbr_magic == htole16(MBR_MAGIC)) {
526 1.1 kiyohara /*
527 1.1 kiyohara * Lookup NetBSD slice. If there is none, go ahead
528 1.1 kiyohara * and try to read the disklabel off sector #0.
529 1.1 kiyohara */
530 1.1 kiyohara mp = mbr->mbr_parts;
531 1.1 kiyohara for (i = 0; i < MBR_PART_COUNT; i++) {
532 1.1 kiyohara if (mp[i].mbrp_type == MBR_PTYPE_NETBSD) {
533 1.1 kiyohara sector = le32toh(mp[i].mbrp_start);
534 1.1 kiyohara break;
535 1.1 kiyohara }
536 1.1 kiyohara }
537 1.1 kiyohara }
538 1.1 kiyohara
539 1.1 kiyohara if (sdstrategy(sd, F_READ, sector + LABELSECTOR, DEV_BSIZE,
540 1.1 kiyohara buf, &rsize))
541 1.1 kiyohara return EOFFSET;
542 1.1 kiyohara
543 1.1 kiyohara msg = getdisklabel((const char *)buf + LABELOFFSET, &sd->sc_label);
544 1.1 kiyohara if (msg)
545 1.1 kiyohara printf("scsi/%d/%d/%d: getdisklabel: %s\n",
546 1.1 kiyohara sd->sc_bus, sd->sc_target, sd->sc_lun, msg);
547 1.1 kiyohara
548 1.1 kiyohara /* check partition */
549 1.1 kiyohara if ((sd->sc_part >= lp->d_npartitions) ||
550 1.1 kiyohara (lp->d_partitions[sd->sc_part].p_fstype == FS_UNUSED)) {
551 1.1 kiyohara DPRINTF(("illegal partition\n"));
552 1.1 kiyohara return EPART;
553 1.1 kiyohara }
554 1.1 kiyohara
555 1.1 kiyohara DPRINTF(("label info: d_secsize %d, d_nsectors %d, d_ncylinders %d,"
556 1.1 kiyohara " d_ntracks %d, d_secpercyl %d\n",
557 1.1 kiyohara sd->sc_label.d_secsize,
558 1.1 kiyohara sd->sc_label.d_nsectors,
559 1.1 kiyohara sd->sc_label.d_ncylinders,
560 1.1 kiyohara sd->sc_label.d_ntracks,
561 1.1 kiyohara sd->sc_label.d_secpercyl));
562 1.1 kiyohara
563 1.1 kiyohara return 0;
564 1.1 kiyohara }
565 1.1 kiyohara
566 1.1 kiyohara /*
567 1.1 kiyohara * Open device (read drive parameters and disklabel)
568 1.1 kiyohara */
569 1.1 kiyohara int
570 1.1 kiyohara sdopen(struct open_file *f, ...)
571 1.1 kiyohara {
572 1.1 kiyohara struct sd_softc *sd;
573 1.1 kiyohara struct scsi_test_unit_ready cmd;
574 1.1 kiyohara struct scsipi_inquiry_data *inqbuf;
575 1.1 kiyohara u_int bus, target, lun, part;
576 1.1 kiyohara int error;
577 1.1 kiyohara char buf[SCSIPI_INQUIRY_LENGTH_SCSI2];
578 1.1 kiyohara va_list ap;
579 1.1 kiyohara
580 1.1 kiyohara va_start(ap, f);
581 1.1 kiyohara bus = va_arg(ap, u_int);
582 1.1 kiyohara target = va_arg(ap, u_int);
583 1.1 kiyohara lun = va_arg(ap, u_int);
584 1.1 kiyohara part = va_arg(ap, u_int);
585 1.1 kiyohara va_end(ap);
586 1.1 kiyohara
587 1.1 kiyohara DPRINTF(("sdopen: scsi/%d/%d/%d_%d\n", bus, target, lun, part));
588 1.1 kiyohara
589 1.1 kiyohara sd = alloc(sizeof(struct sd_softc));
590 1.1 kiyohara if (sd == NULL)
591 1.1 kiyohara return ENOMEM;
592 1.1 kiyohara
593 1.1 kiyohara memset(sd, 0, sizeof(struct sd_softc));
594 1.1 kiyohara
595 1.1 kiyohara sd->sc_part = part;
596 1.1 kiyohara sd->sc_lun = lun;
597 1.1 kiyohara sd->sc_target = target;
598 1.1 kiyohara sd->sc_bus = bus;
599 1.1 kiyohara
600 1.1 kiyohara if ((error = scsi_inquire(sd, sizeof(buf), buf)) != 0)
601 1.1 kiyohara return error;
602 1.1 kiyohara
603 1.1 kiyohara inqbuf = (struct scsipi_inquiry_data *)buf;
604 1.1 kiyohara
605 1.1 kiyohara sd->sc_type = inqbuf->device & SID_TYPE;
606 1.1 kiyohara
607 1.1 kiyohara /*
608 1.1 kiyohara * Determine the operating mode capabilities of the device.
609 1.1 kiyohara */
610 1.1 kiyohara if ((inqbuf->version & SID_ANSII) >= 2) {
611 1.1 kiyohara // if ((inqbuf->flags3 & SID_CmdQue) != 0)
612 1.1 kiyohara // sd->sc_cap |= PERIPH_CAP_TQING;
613 1.1 kiyohara if ((inqbuf->flags3 & SID_Sync) != 0)
614 1.1 kiyohara sd->sc_cap |= PERIPH_CAP_SYNC;
615 1.1 kiyohara
616 1.1 kiyohara /* SPC-2 */
617 1.1 kiyohara if ((inqbuf->version & SID_ANSII) >= 3) {
618 1.1 kiyohara /*
619 1.1 kiyohara * Report ST clocking though CAP_WIDExx/CAP_SYNC.
620 1.1 kiyohara * If the device only supports DT, clear these
621 1.1 kiyohara * flags (DT implies SYNC and WIDE)
622 1.1 kiyohara */
623 1.1 kiyohara switch (inqbuf->flags4 & SID_Clocking) {
624 1.1 kiyohara case SID_CLOCKING_DT_ONLY:
625 1.1 kiyohara sd->sc_cap &= ~PERIPH_CAP_SYNC;
626 1.1 kiyohara break;
627 1.1 kiyohara }
628 1.1 kiyohara }
629 1.1 kiyohara }
630 1.1 kiyohara sd->sc_flags =
631 1.1 kiyohara (inqbuf->dev_qual2 & SID_REMOVABLE) ? FLAGS_REMOVABLE : 0;
632 1.1 kiyohara
633 1.1 kiyohara memset(&cmd, 0, sizeof(cmd));
634 1.1 kiyohara cmd.opcode = SCSI_TEST_UNIT_READY;
635 1.1 kiyohara if ((error = scsi_command(sd, (void *)&cmd, sizeof(cmd), NULL, 0)) != 0)
636 1.1 kiyohara return error;
637 1.1 kiyohara
638 1.1 kiyohara if (sd->sc_flags & FLAGS_REMOVABLE) {
639 1.1 kiyohara printf("XXXXX: removable device found. will not support\n");
640 1.1 kiyohara }
641 1.1 kiyohara if (!(sd->sc_flags & FLAGS_MEDIA_LOADED))
642 1.1 kiyohara sd->sc_flags |= FLAGS_MEDIA_LOADED;
643 1.1 kiyohara
644 1.1 kiyohara if ((error = sd_get_parms(sd)) != 0)
645 1.1 kiyohara return error;
646 1.1 kiyohara
647 1.1 kiyohara strncpy(sd->sc_label.d_typename, inqbuf->product, 16);
648 1.1 kiyohara if ((error = sdgetdisklabel(sd)) != 0)
649 1.1 kiyohara return error;
650 1.1 kiyohara
651 1.1 kiyohara f->f_devdata = sd;
652 1.1 kiyohara return 0;
653 1.1 kiyohara }
654 1.1 kiyohara
655 1.1 kiyohara /*
656 1.1 kiyohara * Close device.
657 1.1 kiyohara */
658 1.1 kiyohara int
659 1.1 kiyohara sdclose(struct open_file *f)
660 1.1 kiyohara {
661 1.1 kiyohara
662 1.1 kiyohara return 0;
663 1.1 kiyohara }
664 1.1 kiyohara
665 1.1 kiyohara /*
666 1.1 kiyohara * Read some data.
667 1.1 kiyohara */
668 1.1 kiyohara int
669 1.1 kiyohara sdstrategy(void *f, int rw, daddr_t dblk, size_t size, void *p, size_t *rsize)
670 1.1 kiyohara {
671 1.1 kiyohara struct sd_softc *sd;
672 1.1 kiyohara struct disklabel *lp;
673 1.1 kiyohara struct partition *pp;
674 1.1 kiyohara struct scsipi_generic *cmdp;
675 1.1 kiyohara struct scsipi_rw_16 cmd16;
676 1.1 kiyohara struct scsipi_rw_10 cmd_big;
677 1.1 kiyohara struct scsi_rw_6 cmd_small;
678 1.1 kiyohara daddr_t blkno;
679 1.1 kiyohara int cmdlen, nsect, i;
680 1.1 kiyohara uint8_t *buf;
681 1.1 kiyohara
682 1.1 kiyohara if (size == 0)
683 1.1 kiyohara return 0;
684 1.1 kiyohara
685 1.1 kiyohara if (rw != F_READ)
686 1.1 kiyohara return EOPNOTSUPP;
687 1.1 kiyohara
688 1.1 kiyohara buf = p;
689 1.1 kiyohara sd = f;
690 1.1 kiyohara lp = &sd->sc_label;
691 1.1 kiyohara pp = &lp->d_partitions[sd->sc_part];
692 1.1 kiyohara
693 1.1 kiyohara if (!(sd->sc_flags & FLAGS_MEDIA_LOADED))
694 1.1 kiyohara return EIO;
695 1.1 kiyohara
696 1.1 kiyohara nsect = howmany(size, lp->d_secsize);
697 1.1 kiyohara blkno = dblk + pp->p_offset;
698 1.1 kiyohara
699 1.1 kiyohara for (i = 0; i < nsect; i++, blkno++) {
700 1.1 kiyohara int error;
701 1.1 kiyohara
702 1.1 kiyohara /*
703 1.1 kiyohara * Fill out the scsi command. Use the smallest CDB possible
704 1.1 kiyohara * (6-byte, 10-byte, or 16-byte).
705 1.1 kiyohara */
706 1.1 kiyohara if ((blkno & 0x1fffff) == blkno) {
707 1.1 kiyohara /* 6-byte CDB */
708 1.1 kiyohara memset(&cmd_small, 0, sizeof(cmd_small));
709 1.1 kiyohara cmd_small.opcode = SCSI_READ_6_COMMAND;
710 1.1 kiyohara _lto3b(blkno, cmd_small.addr);
711 1.1 kiyohara cmd_small.length = 1;
712 1.1 kiyohara cmdlen = sizeof(cmd_small);
713 1.1 kiyohara cmdp = (struct scsipi_generic *)&cmd_small;
714 1.1 kiyohara } else if ((blkno & 0xffffffff) == blkno) {
715 1.1 kiyohara /* 10-byte CDB */
716 1.1 kiyohara memset(&cmd_big, 0, sizeof(cmd_big));
717 1.1 kiyohara cmd_small.opcode = READ_10;
718 1.1 kiyohara _lto4b(blkno, cmd_big.addr);
719 1.1 kiyohara _lto2b(1, cmd_big.length);
720 1.1 kiyohara cmdlen = sizeof(cmd_big);
721 1.1 kiyohara cmdp = (struct scsipi_generic *)&cmd_big;
722 1.1 kiyohara } else {
723 1.1 kiyohara /* 16-byte CDB */
724 1.1 kiyohara memset(&cmd16, 0, sizeof(cmd16));
725 1.1 kiyohara cmd_small.opcode = READ_16;
726 1.1 kiyohara _lto8b(blkno, cmd16.addr);
727 1.1 kiyohara _lto4b(1, cmd16.length);
728 1.1 kiyohara cmdlen = sizeof(cmd16);
729 1.1 kiyohara cmdp = (struct scsipi_generic *)&cmd16;
730 1.1 kiyohara }
731 1.1 kiyohara
732 1.1 kiyohara error = scsi_command(sd, cmdp, cmdlen, buf, lp->d_secsize);
733 1.1 kiyohara if (error)
734 1.1 kiyohara return error;
735 1.1 kiyohara
736 1.1 kiyohara buf += lp->d_secsize;
737 1.1 kiyohara }
738 1.1 kiyohara
739 1.1 kiyohara *rsize = size;
740 1.1 kiyohara return 0;
741 1.1 kiyohara }
742