VirtualBox

source: vbox/trunk/src/VBox/Devices/PC/BIOS/scsi.c@ 42258

Last change on this file since 42258 was 41543, checked in by vboxsync, 13 years ago

BIOS-new: 16-bit/32-bit fix

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 14.2 KB
Line 
1/* $Id: scsi.c 41543 2012-06-01 13:33:40Z vboxsync $ */
2/** @file
3 * SCSI host adapter driver to boot from SCSI disks
4 */
5
6/*
7 * Copyright (C) 2004-2011 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18#include <stdint.h>
19#include <string.h>
20#include "biosint.h"
21#include "inlines.h"
22#include "ebda.h"
23
24
25//#define VBOX_SCSI_DEBUG 1 /* temporary */
26
27#ifdef VBOX_SCSI_DEBUG
28# define VBSCSI_DEBUG(...) BX_INFO(__VA_ARGS__)
29#else
30# define VBSCSI_DEBUG(...)
31#endif
32
33#define VBSCSI_BUSY (1 << 0)
34
35/* The I/O port of the BusLogic SCSI adapter. */
36#define BUSLOGIC_BIOS_IO_PORT 0x330
37/* The I/O port of the LsiLogic SCSI adapter. */
38#define LSILOGIC_BIOS_IO_PORT 0x340
39/* The I/O port of the LsiLogic SAS adapter. */
40#define LSILOGIC_SAS_BIOS_IO_PORT 0x350
41
42#define VBSCSI_REGISTER_STATUS 0
43#define VBSCSI_REGISTER_COMMAND 0
44#define VBSCSI_REGISTER_DATA_IN 1
45#define VBSCSI_REGISTER_IDENTIFY 2
46#define VBSCSI_REGISTER_RESET 3
47
48#define VBSCSI_MAX_DEVICES 16 /* Maximum number of devices a SCSI device can have. */
49
50/* Command opcodes. */
51#define SCSI_INQUIRY 0x12
52#define SCSI_READ_CAPACITY 0x25
53#define SCSI_READ_10 0x28
54#define SCSI_WRITE_10 0x2a
55
56/* Data transfer direction. */
57#define SCSI_TXDIR_FROM_DEVICE 0
58#define SCSI_TXDIR_TO_DEVICE 1
59
60#pragma pack(1)
61
62/* READ_10/WRITE_10 CDB layout. */
63typedef struct {
64 uint16_t command; /* Command. */
65 uint32_t lba; /* LBA, MSB first! */
66 uint8_t pad1; /* Unused. */
67 uint16_t nsect; /* Sector count, MSB first! */
68 uint8_t pad2; /* Unused. */
69} cdb_rw10;
70
71#pragma pack()
72
73ct_assert(sizeof(cdb_rw10) == 10);
74
75int scsi_cmd_data_in(uint16_t io_base, uint8_t device_id, uint8_t __far *aCDB,
76 uint8_t cbCDB, uint8_t __far *buffer, uint16_t cbBuffer)
77{
78 /* Check that the adapter is ready. */
79 uint8_t status;
80 uint16_t i;
81
82 do
83 {
84 status = inb(io_base+VBSCSI_REGISTER_STATUS);
85 } while (status & VBSCSI_BUSY);
86
87 /* Write target ID. */
88 outb(io_base+VBSCSI_REGISTER_COMMAND, device_id);
89 /* Write transfer direction. */
90 outb(io_base+VBSCSI_REGISTER_COMMAND, SCSI_TXDIR_FROM_DEVICE);
91 /* Write the CDB size. */
92 outb(io_base+VBSCSI_REGISTER_COMMAND, cbCDB);
93 /* Write buffer size. */
94 outb(io_base+VBSCSI_REGISTER_COMMAND, cbBuffer);
95 outb(io_base+VBSCSI_REGISTER_COMMAND, (cbBuffer >> 8));
96 /* Write the CDB. */
97 for (i = 0; i < cbCDB; i++)
98 outb(io_base+VBSCSI_REGISTER_COMMAND, aCDB[i]);
99
100 /* Now wait for the command to complete. */
101 do
102 {
103 status = inb(io_base+VBSCSI_REGISTER_STATUS);
104 } while (status & VBSCSI_BUSY);
105
106 /* Get the read data. */
107 rep_insb(buffer, cbBuffer, io_base + VBSCSI_REGISTER_DATA_IN);
108
109 return 0;
110}
111
112int scsi_cmd_data_out(uint16_t io_base, uint8_t device_id, uint8_t __far *aCDB,
113 uint8_t cbCDB, uint8_t __far *buffer, uint16_t cbBuffer)
114{
115 /* Check that the adapter is ready. */
116 uint8_t status;
117 uint16_t i;
118
119 do
120 {
121 status = inb(io_base+VBSCSI_REGISTER_STATUS);
122 } while (status & VBSCSI_BUSY);
123
124 /* Write target ID. */
125 outb(io_base+VBSCSI_REGISTER_COMMAND, device_id);
126 /* Write transfer direction. */
127 outb(io_base+VBSCSI_REGISTER_COMMAND, SCSI_TXDIR_TO_DEVICE);
128 /* Write the CDB size. */
129 outb(io_base+VBSCSI_REGISTER_COMMAND, cbCDB);
130 /* Write buffer size. */
131 outb(io_base+VBSCSI_REGISTER_COMMAND, cbBuffer);
132 outb(io_base+VBSCSI_REGISTER_COMMAND, (cbBuffer >> 8));
133 /* Write the CDB. */
134 for (i = 0; i < cbCDB; i++)
135 outb(io_base+VBSCSI_REGISTER_COMMAND, aCDB[i]);
136
137 /* Write data to I/O port. */
138 rep_outsb(buffer, cbBuffer, io_base+VBSCSI_REGISTER_DATA_IN);
139
140 /* Now wait for the command to complete. */
141 do
142 {
143 status = inb(io_base+VBSCSI_REGISTER_STATUS);
144 } while (status & VBSCSI_BUSY);
145
146 return 0;
147}
148
149/**
150 * Read sectors from an attached SCSI device.
151 *
152 * @returns status code.
153 * @param bios_dsk Pointer to disk request packet (in the
154 * EBDA).
155 */
156int scsi_read_sectors(bio_dsk_t __far *bios_dsk)
157{
158 uint8_t rc;
159 cdb_rw10 cdb;
160 uint16_t count;
161 uint16_t io_base;
162 uint8_t target_id;
163 uint8_t device_id;
164
165 device_id = bios_dsk->drqp.dev_id - BX_MAX_ATA_DEVICES;
166 if (device_id > BX_MAX_SCSI_DEVICES)
167 BX_PANIC("scsi_read_sectors: device_id out of range %d\n", device_id);
168
169 count = bios_dsk->drqp.nsect;
170
171 /* Prepare a CDB. */
172 cdb.command = SCSI_READ_10;
173 cdb.lba = swap_32(bios_dsk->drqp.lba);
174 cdb.pad1 = 0;
175 cdb.nsect = swap_16(count);
176 cdb.pad2 = 0;
177
178
179 io_base = bios_dsk->scsidev[device_id].io_base;
180 target_id = bios_dsk->scsidev[device_id].target_id;
181
182 rc = scsi_cmd_data_in(io_base, target_id, (void __far *)&cdb, 10,
183 bios_dsk->drqp.buffer, (count * 512));
184
185 if (!rc)
186 {
187 bios_dsk->drqp.trsfsectors = count;
188 bios_dsk->drqp.trsfbytes = count * 512;
189 }
190
191 return rc;
192}
193
194/**
195 * Write sectors to an attached SCSI device.
196 *
197 * @returns status code.
198 * @param bios_dsk Pointer to disk request packet (in the
199 * EBDA).
200 */
201int scsi_write_sectors(bio_dsk_t __far *bios_dsk)
202{
203 uint8_t rc;
204 cdb_rw10 cdb;
205 uint16_t count;
206 uint16_t io_base;
207 uint8_t target_id;
208 uint8_t device_id;
209
210 device_id = bios_dsk->drqp.dev_id - BX_MAX_ATA_DEVICES;
211 if (device_id > BX_MAX_SCSI_DEVICES)
212 BX_PANIC("scsi_write_sectors: device_id out of range %d\n", device_id);
213
214 count = bios_dsk->drqp.nsect;
215
216 /* Prepare a CDB. */
217 cdb.command = SCSI_WRITE_10;
218 cdb.lba = swap_32(bios_dsk->drqp.lba);
219 cdb.pad1 = 0;
220 cdb.nsect = swap_16(count);
221 cdb.pad2 = 0;
222
223 io_base = bios_dsk->scsidev[device_id].io_base;
224 target_id = bios_dsk->scsidev[device_id].target_id;
225
226 rc = scsi_cmd_data_out(io_base, target_id, (void __far *)&cdb, 10,
227 bios_dsk->drqp.buffer, (count * 512));
228
229 if (!rc)
230 {
231 bios_dsk->drqp.trsfsectors = count;
232 bios_dsk->drqp.trsfbytes = (count * 512);
233 }
234
235 return rc;
236}
237
238/**
239 * Enumerate attached devices.
240 *
241 * @returns nothing.
242 * @param io_base The I/O base port of the controller.
243 */
244void scsi_enumerate_attached_devices(uint16_t io_base)
245{
246 int i;
247 uint8_t buffer[0x0200];
248 bio_dsk_t __far *bios_dsk;
249
250 bios_dsk = read_word(0x0040, 0x000E) :> &EbdaData->bdisk;
251
252 /* Go through target devices. */
253 for (i = 0; i < VBSCSI_MAX_DEVICES; i++)
254 {
255 uint8_t rc;
256 uint8_t aCDB[10];
257
258 aCDB[0] = SCSI_INQUIRY;
259 aCDB[1] = 0;
260 aCDB[2] = 0;
261 aCDB[3] = 0;
262 aCDB[4] = 5; /* Allocation length. */
263 aCDB[5] = 0;
264
265 rc = scsi_cmd_data_in(io_base, i, aCDB, 6, buffer, 5);
266 if (rc != 0)
267 BX_PANIC("scsi_enumerate_attached_devices: SCSI_INQUIRY failed\n");
268
269 /* Check if there is a disk attached. */
270 if ( ((buffer[0] & 0xe0) == 0)
271 && ((buffer[0] & 0x1f) == 0x00))
272 {
273 VBSCSI_DEBUG("scsi_enumerate_attached_devices: Disk detected at %d\n", i);
274
275 /* We add the disk only if the maximum is not reached yet. */
276 if (bios_dsk->scsi_hdcount < BX_MAX_SCSI_DEVICES)
277 {
278 uint32_t sectors, sector_size, cylinders;
279 uint16_t heads, sectors_per_track;
280 uint8_t hdcount, hdcount_scsi, hd_index;
281
282 /* Issue a read capacity command now. */
283 _fmemset(aCDB, 0, sizeof(aCDB));
284 aCDB[0] = SCSI_READ_CAPACITY;
285
286 rc = scsi_cmd_data_in(io_base, i, aCDB, 10, buffer, 8);
287 if (rc != 0)
288 BX_PANIC("scsi_enumerate_attached_devices: SCSI_READ_CAPACITY failed\n");
289
290 /* Build sector number and size from the buffer. */
291 //@todo: byte swapping for dword sized items should be farmed out...
292 sectors = ((uint32_t)buffer[0] << 24)
293 | ((uint32_t)buffer[1] << 16)
294 | ((uint32_t)buffer[2] << 8)
295 | ((uint32_t)buffer[3]);
296
297 sector_size = ((uint32_t)buffer[4] << 24)
298 | ((uint32_t)buffer[5] << 16)
299 | ((uint32_t)buffer[6] << 8)
300 | ((uint32_t)buffer[7]);
301
302 /* We only support the disk if sector size is 512 bytes. */
303 if (sector_size != 512)
304 {
305 /* Leave a log entry. */
306 BX_INFO("Disk %d has an unsupported sector size of %u\n", i, sector_size);
307 continue;
308 }
309
310 /* We need to calculate the geometry for the disk. From
311 * the BusLogic driver in the Linux kernel.
312 */
313 if (sectors >= (uint32_t)4 * 1024 * 1024)
314 {
315 heads = 255;
316 sectors_per_track = 63;
317 }
318 else if (sectors >= (uint32_t)2 * 1024 * 1024)
319 {
320 heads = 128;
321 sectors_per_track = 32;
322 }
323 else
324 {
325 heads = 64;
326 sectors_per_track = 32;
327 }
328 cylinders = (uint32_t)(sectors / (heads * sectors_per_track));
329 hdcount_scsi = bios_dsk->scsi_hdcount;
330
331 /* Calculate index into the generic disk table. */
332 hd_index = hdcount_scsi + BX_MAX_ATA_DEVICES;
333
334 bios_dsk->scsidev[hdcount_scsi].io_base = io_base;
335 bios_dsk->scsidev[hdcount_scsi].target_id = i;
336 bios_dsk->devices[hd_index].type = DSK_TYPE_SCSI;
337 bios_dsk->devices[hd_index].device = DSK_DEVICE_HD;
338 bios_dsk->devices[hd_index].removable = 0;
339 bios_dsk->devices[hd_index].lock = 0;
340 bios_dsk->devices[hd_index].blksize = sector_size;
341 bios_dsk->devices[hd_index].translation = GEO_TRANSLATION_LBA;
342
343 /* Write LCHS values. */
344 bios_dsk->devices[hd_index].lchs.heads = heads;
345 bios_dsk->devices[hd_index].lchs.spt = sectors_per_track;
346 if (cylinders > 1024)
347 bios_dsk->devices[hd_index].lchs.cylinders = 1024;
348 else
349 bios_dsk->devices[hd_index].lchs.cylinders = (uint16_t)cylinders;
350
351 /* Write PCHS values. */
352 bios_dsk->devices[hd_index].pchs.heads = heads;
353 bios_dsk->devices[hd_index].pchs.spt = sectors_per_track;
354 if (cylinders > 1024)
355 bios_dsk->devices[hd_index].pchs.cylinders = 1024;
356 else
357 bios_dsk->devices[hd_index].pchs.cylinders = (uint16_t)cylinders;
358
359 bios_dsk->devices[hd_index].sectors = sectors;
360
361 /* Store the id of the disk in the ata hdidmap. */
362 hdcount = bios_dsk->hdcount;
363 bios_dsk->hdidmap[hdcount] = hdcount_scsi + BX_MAX_ATA_DEVICES;
364 hdcount++;
365 bios_dsk->hdcount = hdcount;
366
367 /* Update hdcount in the BDA. */
368 hdcount = read_byte(0x40, 0x75);
369 hdcount++;
370 write_byte(0x40, 0x75, hdcount);
371
372 hdcount_scsi++;
373 bios_dsk->scsi_hdcount = hdcount_scsi;
374 }
375 else
376 {
377 /* We reached the maximum of SCSI disks we can boot from. We can quit detecting. */
378 break;
379 }
380 }
381 else
382 VBSCSI_DEBUG("scsi_enumerate_attached_devices: No disk detected at %d\n", i);
383 }
384}
385
386/**
387 * Init the SCSI driver and detect attached disks.
388 */
389void BIOSCALL scsi_init(void)
390{
391 uint8_t identifier;
392 bio_dsk_t __far *bios_dsk;
393
394 bios_dsk = read_word(0x0040, 0x000E) :> &EbdaData->bdisk;
395
396 bios_dsk->scsi_hdcount = 0;
397
398 identifier = 0;
399
400 /* Detect BusLogic adapter. */
401 outb(BUSLOGIC_BIOS_IO_PORT+VBSCSI_REGISTER_IDENTIFY, 0x55);
402 identifier = inb(BUSLOGIC_BIOS_IO_PORT+VBSCSI_REGISTER_IDENTIFY);
403
404 if (identifier == 0x55)
405 {
406 /* Detected - Enumerate attached devices. */
407 VBSCSI_DEBUG("scsi_init: BusLogic SCSI adapter detected\n");
408 outb(BUSLOGIC_BIOS_IO_PORT+VBSCSI_REGISTER_RESET, 0);
409 scsi_enumerate_attached_devices(BUSLOGIC_BIOS_IO_PORT);
410 }
411 else
412 {
413 VBSCSI_DEBUG("scsi_init: BusLogic SCSI adapter not detected\n");
414 }
415
416 /* Detect LsiLogic adapter. */
417 outb(LSILOGIC_BIOS_IO_PORT+VBSCSI_REGISTER_IDENTIFY, 0x55);
418 identifier = inb(LSILOGIC_BIOS_IO_PORT+VBSCSI_REGISTER_IDENTIFY);
419
420 if (identifier == 0x55)
421 {
422 /* Detected - Enumerate attached devices. */
423 VBSCSI_DEBUG("scsi_init: LSI Logic SCSI adapter detected\n");
424 outb(LSILOGIC_BIOS_IO_PORT+VBSCSI_REGISTER_RESET, 0);
425 scsi_enumerate_attached_devices(LSILOGIC_BIOS_IO_PORT);
426 }
427 else
428 {
429 VBSCSI_DEBUG("scsi_init: LSI Logic SCSI adapter not detected\n");
430 }
431
432 /* Detect LsiLogic SAS adapter. */
433 outb(LSILOGIC_SAS_BIOS_IO_PORT+VBSCSI_REGISTER_IDENTIFY, 0x55);
434 identifier = inb(LSILOGIC_SAS_BIOS_IO_PORT+VBSCSI_REGISTER_IDENTIFY);
435
436 if (identifier == 0x55)
437 {
438 /* Detected - Enumerate attached devices. */
439 VBSCSI_DEBUG("scsi_init: LSI Logic SAS adapter detected\n");
440 outb(LSILOGIC_SAS_BIOS_IO_PORT+VBSCSI_REGISTER_RESET, 0);
441 scsi_enumerate_attached_devices(LSILOGIC_SAS_BIOS_IO_PORT);
442 }
443 else
444 {
445 VBSCSI_DEBUG("scsi_init: LSI Logic SAS adapter not detected\n");
446 }
447}
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette