17c478bd9Sstevel@tonic-gate /* 27c478bd9Sstevel@tonic-gate * CDDL HEADER START 37c478bd9Sstevel@tonic-gate * 47c478bd9Sstevel@tonic-gate * The contents of this file are subject to the terms of the 57c478bd9Sstevel@tonic-gate * Common Development and Distribution License, Version 1.0 only 67c478bd9Sstevel@tonic-gate * (the "License"). You may not use this file except in compliance 77c478bd9Sstevel@tonic-gate * with the License. 87c478bd9Sstevel@tonic-gate * 97c478bd9Sstevel@tonic-gate * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 107c478bd9Sstevel@tonic-gate * or http://www.opensolaris.org/os/licensing. 117c478bd9Sstevel@tonic-gate * See the License for the specific language governing permissions 127c478bd9Sstevel@tonic-gate * and limitations under the License. 137c478bd9Sstevel@tonic-gate * 147c478bd9Sstevel@tonic-gate * When distributing Covered Code, include this CDDL HEADER in each 157c478bd9Sstevel@tonic-gate * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 167c478bd9Sstevel@tonic-gate * If applicable, add the following below this CDDL HEADER, with the 177c478bd9Sstevel@tonic-gate * fields enclosed by brackets "[]" replaced with your own identifying 187c478bd9Sstevel@tonic-gate * information: Portions Copyright [yyyy] [name of copyright owner] 197c478bd9Sstevel@tonic-gate * 207c478bd9Sstevel@tonic-gate * CDDL HEADER END 217c478bd9Sstevel@tonic-gate */ 227c478bd9Sstevel@tonic-gate /* 23*dc36087fSec158148 * Copyright 2005 Sun Microsystems, Inc. All rights reserved. 247c478bd9Sstevel@tonic-gate * Use is subject to license terms. 257c478bd9Sstevel@tonic-gate */ 267c478bd9Sstevel@tonic-gate 277c478bd9Sstevel@tonic-gate #pragma ident "%Z%%M% %I% %E% SMI" 287c478bd9Sstevel@tonic-gate 297c478bd9Sstevel@tonic-gate #include <sys/types.h> 307c478bd9Sstevel@tonic-gate #include <stdlib.h> 317c478bd9Sstevel@tonic-gate #include <stdio.h> 327c478bd9Sstevel@tonic-gate #include <string.h> 337c478bd9Sstevel@tonic-gate 347c478bd9Sstevel@tonic-gate #include "transport.h" 357c478bd9Sstevel@tonic-gate #include "mmc.h" 367c478bd9Sstevel@tonic-gate #include "util.h" 377c478bd9Sstevel@tonic-gate #include "main.h" 387c478bd9Sstevel@tonic-gate 397c478bd9Sstevel@tonic-gate int 407c478bd9Sstevel@tonic-gate test_unit_ready(int fd) 417c478bd9Sstevel@tonic-gate { 427c478bd9Sstevel@tonic-gate struct uscsi_cmd *scmd; 437c478bd9Sstevel@tonic-gate 447c478bd9Sstevel@tonic-gate scmd = get_uscsi_cmd(); 457c478bd9Sstevel@tonic-gate scmd->uscsi_flags = USCSI_SILENT; 467c478bd9Sstevel@tonic-gate scmd->uscsi_timeout = DEFAULT_SCSI_TIMEOUT; 477c478bd9Sstevel@tonic-gate /* give length of cdb structure */ 487c478bd9Sstevel@tonic-gate scmd->uscsi_cdblen = 6; 497c478bd9Sstevel@tonic-gate if ((uscsi_error = uscsi(fd, scmd)) < 0) 507c478bd9Sstevel@tonic-gate return (0); 517c478bd9Sstevel@tonic-gate return (1); 527c478bd9Sstevel@tonic-gate } 537c478bd9Sstevel@tonic-gate 547c478bd9Sstevel@tonic-gate int 557c478bd9Sstevel@tonic-gate inquiry(int fd, uchar_t *inq) 567c478bd9Sstevel@tonic-gate { 577c478bd9Sstevel@tonic-gate struct uscsi_cmd *scmd; 587c478bd9Sstevel@tonic-gate 597c478bd9Sstevel@tonic-gate scmd = get_uscsi_cmd(); 607c478bd9Sstevel@tonic-gate scmd->uscsi_flags = USCSI_READ|USCSI_SILENT; 617c478bd9Sstevel@tonic-gate scmd->uscsi_timeout = DEFAULT_SCSI_TIMEOUT; 627c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[0] = INQUIRY_CMD; 637c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[4] = INQUIRY_DATA_LENGTH; 647c478bd9Sstevel@tonic-gate scmd->uscsi_cdblen = 6; 657c478bd9Sstevel@tonic-gate scmd->uscsi_bufaddr = (char *)inq; 667c478bd9Sstevel@tonic-gate scmd->uscsi_buflen = INQUIRY_DATA_LENGTH; 677c478bd9Sstevel@tonic-gate if ((uscsi_error = uscsi(fd, scmd)) < 0) 687c478bd9Sstevel@tonic-gate return (0); 697c478bd9Sstevel@tonic-gate return (1); 707c478bd9Sstevel@tonic-gate } 717c478bd9Sstevel@tonic-gate 727c478bd9Sstevel@tonic-gate int 737c478bd9Sstevel@tonic-gate read_capacity(int fd, uchar_t *capbuf) 747c478bd9Sstevel@tonic-gate { 757c478bd9Sstevel@tonic-gate struct uscsi_cmd *scmd; 767c478bd9Sstevel@tonic-gate 777c478bd9Sstevel@tonic-gate scmd = get_uscsi_cmd(); 787c478bd9Sstevel@tonic-gate scmd->uscsi_flags = USCSI_READ|USCSI_SILENT; 797c478bd9Sstevel@tonic-gate scmd->uscsi_timeout = DEFAULT_SCSI_TIMEOUT; 807c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[0] = READ_CAP_CMD; 817c478bd9Sstevel@tonic-gate scmd->uscsi_cdblen = 10; 827c478bd9Sstevel@tonic-gate scmd->uscsi_bufaddr = (char *)capbuf; 837c478bd9Sstevel@tonic-gate scmd->uscsi_buflen = 8; 847c478bd9Sstevel@tonic-gate if ((uscsi_error = uscsi(fd, scmd)) < 0) 857c478bd9Sstevel@tonic-gate return (0); 867c478bd9Sstevel@tonic-gate return (1); 877c478bd9Sstevel@tonic-gate } 887c478bd9Sstevel@tonic-gate 897c478bd9Sstevel@tonic-gate int 907c478bd9Sstevel@tonic-gate mode_sense(int fd, uchar_t pc, int dbd, int page_len, uchar_t *buffer) 917c478bd9Sstevel@tonic-gate { 927c478bd9Sstevel@tonic-gate struct uscsi_cmd *scmd; 937c478bd9Sstevel@tonic-gate 947c478bd9Sstevel@tonic-gate scmd = get_uscsi_cmd(); 957c478bd9Sstevel@tonic-gate scmd->uscsi_flags = USCSI_READ|USCSI_SILENT; 967c478bd9Sstevel@tonic-gate scmd->uscsi_buflen = page_len; 977c478bd9Sstevel@tonic-gate scmd->uscsi_bufaddr = (char *)buffer; 987c478bd9Sstevel@tonic-gate scmd->uscsi_timeout = DEFAULT_SCSI_TIMEOUT; 997c478bd9Sstevel@tonic-gate scmd->uscsi_cdblen = 0xa; 1007c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[0] = MODE_SENSE_10_CMD; 1017c478bd9Sstevel@tonic-gate if (dbd) { 1027c478bd9Sstevel@tonic-gate /* don't return any block descriptors */ 1037c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[1] = 0x8; 1047c478bd9Sstevel@tonic-gate } 1057c478bd9Sstevel@tonic-gate /* the page code we want */ 1067c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[2] = pc; 1077c478bd9Sstevel@tonic-gate /* allocation length */ 1087c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[7] = (page_len >> 8) & 0xff; 1097c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[8] = page_len & 0xff; 1107c478bd9Sstevel@tonic-gate 1117c478bd9Sstevel@tonic-gate if ((uscsi_error = uscsi(fd, scmd)) < 0) 1127c478bd9Sstevel@tonic-gate return (0); 1137c478bd9Sstevel@tonic-gate return (1); 1147c478bd9Sstevel@tonic-gate } 1157c478bd9Sstevel@tonic-gate 1167c478bd9Sstevel@tonic-gate int 1177c478bd9Sstevel@tonic-gate mode_select(int fd, int page_len, uchar_t *buffer) 1187c478bd9Sstevel@tonic-gate { 1197c478bd9Sstevel@tonic-gate struct uscsi_cmd *scmd; 1207c478bd9Sstevel@tonic-gate 1217c478bd9Sstevel@tonic-gate scmd = get_uscsi_cmd(); 1227c478bd9Sstevel@tonic-gate scmd->uscsi_flags = USCSI_WRITE|USCSI_SILENT; 1237c478bd9Sstevel@tonic-gate scmd->uscsi_buflen = page_len; 1247c478bd9Sstevel@tonic-gate scmd->uscsi_bufaddr = (char *)buffer; 1257c478bd9Sstevel@tonic-gate scmd->uscsi_timeout = DEFAULT_SCSI_TIMEOUT; 1267c478bd9Sstevel@tonic-gate scmd->uscsi_cdblen = 0xa; 1277c478bd9Sstevel@tonic-gate 1287c478bd9Sstevel@tonic-gate /* mode select (10) command */ 1297c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[0] = MODE_SELECT_10_CMD; 1307c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[1] = 0x10; 1317c478bd9Sstevel@tonic-gate 1327c478bd9Sstevel@tonic-gate /* parameter list length */ 1337c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[7] = (page_len >> 8) & 0xff; 1347c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[8] = page_len & 0xff; 1357c478bd9Sstevel@tonic-gate 1367c478bd9Sstevel@tonic-gate if ((uscsi_error = uscsi(fd, scmd)) < 0) 1377c478bd9Sstevel@tonic-gate return (0); 1387c478bd9Sstevel@tonic-gate return (1); 1397c478bd9Sstevel@tonic-gate } 1407c478bd9Sstevel@tonic-gate 1417c478bd9Sstevel@tonic-gate int 1427c478bd9Sstevel@tonic-gate read_track_info(int fd, int trackno, uchar_t *ti) 1437c478bd9Sstevel@tonic-gate { 1447c478bd9Sstevel@tonic-gate struct uscsi_cmd *scmd; 1457c478bd9Sstevel@tonic-gate 1467c478bd9Sstevel@tonic-gate scmd = get_uscsi_cmd(); 1477c478bd9Sstevel@tonic-gate scmd->uscsi_flags = USCSI_READ|USCSI_SILENT; 1487c478bd9Sstevel@tonic-gate scmd->uscsi_timeout = DEFAULT_SCSI_TIMEOUT; 1497c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[0] = READ_TRACK_CMD; 1507c478bd9Sstevel@tonic-gate 1517c478bd9Sstevel@tonic-gate /* tell device we are giving it a track number */ 1527c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[1] = 1; 1537c478bd9Sstevel@tonic-gate 1547c478bd9Sstevel@tonic-gate /* track number to read */ 1557c478bd9Sstevel@tonic-gate if (trackno == -1) 1567c478bd9Sstevel@tonic-gate if (device_type == CD_RW) { 1577c478bd9Sstevel@tonic-gate ((uchar_t *)scmd->uscsi_cdb)[5] = 0xff; 1587c478bd9Sstevel@tonic-gate } else { 1597c478bd9Sstevel@tonic-gate /* only 1 track is allowed on DVD media */ 1607c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[1] = 0; 1617c478bd9Sstevel@tonic-gate ((uchar_t *)scmd->uscsi_cdb)[5] = 0; 1627c478bd9Sstevel@tonic-gate } 1637c478bd9Sstevel@tonic-gate else 1647c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[5] = (uchar_t)trackno; 1657c478bd9Sstevel@tonic-gate 1667c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[8] = TRACK_INFO_SIZE; 1677c478bd9Sstevel@tonic-gate scmd->uscsi_cdblen = 10; 1687c478bd9Sstevel@tonic-gate scmd->uscsi_bufaddr = (char *)ti; 1697c478bd9Sstevel@tonic-gate scmd->uscsi_buflen = TRACK_INFO_SIZE; 1707c478bd9Sstevel@tonic-gate if ((uscsi_error = uscsi(fd, scmd)) < 0) 1717c478bd9Sstevel@tonic-gate return (0); 1727c478bd9Sstevel@tonic-gate return (1); 1737c478bd9Sstevel@tonic-gate } 1747c478bd9Sstevel@tonic-gate 1757c478bd9Sstevel@tonic-gate int 1767c478bd9Sstevel@tonic-gate read_toc(int fd, int format, int trackno, int buflen, uchar_t *buf) 1777c478bd9Sstevel@tonic-gate { 1787c478bd9Sstevel@tonic-gate struct uscsi_cmd *scmd; 1797c478bd9Sstevel@tonic-gate 1807c478bd9Sstevel@tonic-gate scmd = get_uscsi_cmd(); 1817c478bd9Sstevel@tonic-gate scmd->uscsi_flags = USCSI_READ|USCSI_SILENT; 1827c478bd9Sstevel@tonic-gate scmd->uscsi_timeout = DEFAULT_SCSI_TIMEOUT; 1837c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[0] = READ_TOC_CMD; 1847c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[2] = format & 0xf; 1857c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[6] = trackno; 1867c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[8] = buflen & 0xff; 1877c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[7] = (buflen >> 8) & 0xff; 1887c478bd9Sstevel@tonic-gate scmd->uscsi_cdblen = 10; 1897c478bd9Sstevel@tonic-gate scmd->uscsi_bufaddr = (char *)buf; 1907c478bd9Sstevel@tonic-gate scmd->uscsi_buflen = buflen; 1917c478bd9Sstevel@tonic-gate if ((uscsi_error = uscsi(fd, scmd)) < 0) 1927c478bd9Sstevel@tonic-gate return (0); 1937c478bd9Sstevel@tonic-gate 1947c478bd9Sstevel@tonic-gate /* Fix for old SONY drives */ 1957c478bd9Sstevel@tonic-gate if ((format == 0) && (buflen == 4) && (buf[0] == 0) && (buf[1] == 2)) { 1967c478bd9Sstevel@tonic-gate uint16_t toc_size; 1977c478bd9Sstevel@tonic-gate 1987c478bd9Sstevel@tonic-gate toc_size = (((uint16_t)(buf[3] + 1)) * 8) + 2; 1997c478bd9Sstevel@tonic-gate load_scsi16(buf, toc_size); 2007c478bd9Sstevel@tonic-gate } 2017c478bd9Sstevel@tonic-gate return (1); 2027c478bd9Sstevel@tonic-gate } 2037c478bd9Sstevel@tonic-gate 2047c478bd9Sstevel@tonic-gate int 2057c478bd9Sstevel@tonic-gate read_header(int fd, uint32_t lba, uchar_t *buf) 2067c478bd9Sstevel@tonic-gate { 2077c478bd9Sstevel@tonic-gate struct uscsi_cmd *scmd; 2087c478bd9Sstevel@tonic-gate 2097c478bd9Sstevel@tonic-gate scmd = get_uscsi_cmd(); 2107c478bd9Sstevel@tonic-gate scmd->uscsi_flags = USCSI_READ|USCSI_SILENT; 2117c478bd9Sstevel@tonic-gate scmd->uscsi_timeout = DEFAULT_SCSI_TIMEOUT; 2127c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[0] = READ_HDR_CMD; 2137c478bd9Sstevel@tonic-gate 2147c478bd9Sstevel@tonic-gate /* Logical block address */ 2157c478bd9Sstevel@tonic-gate load_scsi32(&scmd->uscsi_cdb[2], lba); 2167c478bd9Sstevel@tonic-gate 2177c478bd9Sstevel@tonic-gate /* allocation length */ 2187c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[8] = 8; 2197c478bd9Sstevel@tonic-gate scmd->uscsi_cdblen = 10; 2207c478bd9Sstevel@tonic-gate scmd->uscsi_bufaddr = (char *)buf; 2217c478bd9Sstevel@tonic-gate scmd->uscsi_buflen = 8; 2227c478bd9Sstevel@tonic-gate if ((uscsi_error = uscsi(fd, scmd)) < 0) 2237c478bd9Sstevel@tonic-gate return (0); 2247c478bd9Sstevel@tonic-gate return (1); 2257c478bd9Sstevel@tonic-gate } 2267c478bd9Sstevel@tonic-gate 2277c478bd9Sstevel@tonic-gate int 2287c478bd9Sstevel@tonic-gate read_disc_info(int fd, uchar_t *di) 2297c478bd9Sstevel@tonic-gate { 2307c478bd9Sstevel@tonic-gate struct uscsi_cmd *scmd; 2317c478bd9Sstevel@tonic-gate 2327c478bd9Sstevel@tonic-gate scmd = get_uscsi_cmd(); 2337c478bd9Sstevel@tonic-gate scmd->uscsi_flags = USCSI_READ|USCSI_SILENT; 2347c478bd9Sstevel@tonic-gate scmd->uscsi_timeout = DEFAULT_SCSI_TIMEOUT; 2357c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[0] = READ_INFO_CMD; 2367c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[8] = DISC_INFO_BLOCK_SIZE; 2377c478bd9Sstevel@tonic-gate scmd->uscsi_cdblen = 10; 2387c478bd9Sstevel@tonic-gate scmd->uscsi_bufaddr = (char *)di; 2397c478bd9Sstevel@tonic-gate scmd->uscsi_buflen = DISC_INFO_BLOCK_SIZE; 2407c478bd9Sstevel@tonic-gate if ((uscsi_error = uscsi(fd, scmd)) < 0) 2417c478bd9Sstevel@tonic-gate return (0); 2427c478bd9Sstevel@tonic-gate return (1); 2437c478bd9Sstevel@tonic-gate } 2447c478bd9Sstevel@tonic-gate 245*dc36087fSec158148 /* Get information about the Logical Unit's capabilities */ 2467c478bd9Sstevel@tonic-gate int 2477c478bd9Sstevel@tonic-gate get_configuration(int fd, uint16_t feature, int bufsize, uchar_t *buf) 2487c478bd9Sstevel@tonic-gate { 2497c478bd9Sstevel@tonic-gate struct uscsi_cmd *scmd; 2507c478bd9Sstevel@tonic-gate 2517c478bd9Sstevel@tonic-gate scmd = get_uscsi_cmd(); 2527c478bd9Sstevel@tonic-gate scmd->uscsi_flags = USCSI_READ|USCSI_SILENT; 2537c478bd9Sstevel@tonic-gate scmd->uscsi_timeout = DEFAULT_SCSI_TIMEOUT; 254*dc36087fSec158148 255*dc36087fSec158148 /* Set OPERATION CODE in CDB */ 2567c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[0] = GET_CONFIG_CMD; 257*dc36087fSec158148 258*dc36087fSec158148 /* 259*dc36087fSec158148 * Set RT field in CDB, currently need at most one 260*dc36087fSec158148 * Feature Descriptor 261*dc36087fSec158148 */ 2627c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[1] = 0x2; 263*dc36087fSec158148 264*dc36087fSec158148 /* Set Starting Feature Number in CDB */ 2657c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[2] = (feature >> 8) & 0xff; 2667c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[3] = feature & 0xff; 267*dc36087fSec158148 268*dc36087fSec158148 /* Set Allocation Length in CDB */ 2697c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[7] = (bufsize >> 8) & 0xff; 2707c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[8] = bufsize & 0xff; 271*dc36087fSec158148 2727c478bd9Sstevel@tonic-gate scmd->uscsi_cdblen = 10; 2737c478bd9Sstevel@tonic-gate scmd->uscsi_bufaddr = (char *)buf; 2747c478bd9Sstevel@tonic-gate scmd->uscsi_buflen = bufsize; 2757c478bd9Sstevel@tonic-gate if ((uscsi_error = uscsi(fd, scmd)) < 0) 2767c478bd9Sstevel@tonic-gate return (0); 2777c478bd9Sstevel@tonic-gate return (1); 2787c478bd9Sstevel@tonic-gate } 2797c478bd9Sstevel@tonic-gate 2807c478bd9Sstevel@tonic-gate int 2817c478bd9Sstevel@tonic-gate read10(int fd, uint32_t start_blk, uint16_t nblk, uchar_t *buf, 2827c478bd9Sstevel@tonic-gate uint32_t bufsize) 2837c478bd9Sstevel@tonic-gate { 2847c478bd9Sstevel@tonic-gate struct uscsi_cmd *scmd; 2857c478bd9Sstevel@tonic-gate 2867c478bd9Sstevel@tonic-gate scmd = get_uscsi_cmd(); 2877c478bd9Sstevel@tonic-gate scmd->uscsi_flags = USCSI_READ|USCSI_SILENT; 2887c478bd9Sstevel@tonic-gate scmd->uscsi_timeout = DEFAULT_SCSI_TIMEOUT; 2897c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[0] = READ_10_CMD; 2907c478bd9Sstevel@tonic-gate load_scsi32(&scmd->uscsi_cdb[2], start_blk); 2917c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[8] = nblk & 0xff; 2927c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[7] = (nblk >> 8) & 0xff; 2937c478bd9Sstevel@tonic-gate scmd->uscsi_cdblen = 10; 2947c478bd9Sstevel@tonic-gate scmd->uscsi_bufaddr = (char *)buf; 2957c478bd9Sstevel@tonic-gate scmd->uscsi_buflen = bufsize; 2967c478bd9Sstevel@tonic-gate if ((uscsi_error = uscsi(fd, scmd)) < 0) 2977c478bd9Sstevel@tonic-gate return (0); 2987c478bd9Sstevel@tonic-gate return (1); 2997c478bd9Sstevel@tonic-gate } 3007c478bd9Sstevel@tonic-gate 3017c478bd9Sstevel@tonic-gate int 3027c478bd9Sstevel@tonic-gate write10(int fd, uint32_t start_blk, uint16_t nblk, uchar_t *buf, 3037c478bd9Sstevel@tonic-gate uint32_t bufsize) 3047c478bd9Sstevel@tonic-gate { 3057c478bd9Sstevel@tonic-gate struct uscsi_cmd *scmd; 3067c478bd9Sstevel@tonic-gate 3077c478bd9Sstevel@tonic-gate scmd = get_uscsi_cmd(); 3087c478bd9Sstevel@tonic-gate scmd->uscsi_flags = USCSI_WRITE|USCSI_SILENT; 3097c478bd9Sstevel@tonic-gate /* 3107c478bd9Sstevel@tonic-gate * Some DVD drives take longer to write than 3117c478bd9Sstevel@tonic-gate * the standard time, since they tend to generate 3127c478bd9Sstevel@tonic-gate * the media TOC on the fly when the cache is full 3137c478bd9Sstevel@tonic-gate */ 3147c478bd9Sstevel@tonic-gate scmd->uscsi_timeout = DEFAULT_SCSI_TIMEOUT * 3; 3157c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[0] = WRITE_10_CMD; 3167c478bd9Sstevel@tonic-gate load_scsi32(&scmd->uscsi_cdb[2], start_blk); 3177c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[8] = nblk & 0xff; 3187c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[7] = (nblk >> 8) & 0xff; 3197c478bd9Sstevel@tonic-gate scmd->uscsi_cdblen = 10; 3207c478bd9Sstevel@tonic-gate scmd->uscsi_bufaddr = (char *)buf; 3217c478bd9Sstevel@tonic-gate scmd->uscsi_buflen = bufsize; 3227c478bd9Sstevel@tonic-gate if ((uscsi_error = uscsi(fd, scmd)) < 0) 3237c478bd9Sstevel@tonic-gate return (0); 3247c478bd9Sstevel@tonic-gate return (1); 3257c478bd9Sstevel@tonic-gate } 3267c478bd9Sstevel@tonic-gate 3277c478bd9Sstevel@tonic-gate int 3287c478bd9Sstevel@tonic-gate close_track(int fd, int trackno, int close_session, int immediate) 3297c478bd9Sstevel@tonic-gate { 3307c478bd9Sstevel@tonic-gate struct uscsi_cmd *scmd; 3317c478bd9Sstevel@tonic-gate 3327c478bd9Sstevel@tonic-gate scmd = get_uscsi_cmd(); 3337c478bd9Sstevel@tonic-gate scmd->uscsi_flags = USCSI_SILENT; 3347c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[0] = CLOSE_TRACK_CMD; 3357c478bd9Sstevel@tonic-gate if (immediate) { 3367c478bd9Sstevel@tonic-gate scmd->uscsi_timeout = DEFAULT_SCSI_TIMEOUT; 3377c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[1] = 1; 3387c478bd9Sstevel@tonic-gate } else { 3397c478bd9Sstevel@tonic-gate scmd->uscsi_timeout = 240; 3407c478bd9Sstevel@tonic-gate } 3417c478bd9Sstevel@tonic-gate if ((close_session) || (device_type == DVD_PLUS) || 3427c478bd9Sstevel@tonic-gate (device_type == DVD_PLUS_W)) { 3437c478bd9Sstevel@tonic-gate /* close the session */ 3447c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[2] = 2; 3457c478bd9Sstevel@tonic-gate 3467c478bd9Sstevel@tonic-gate } else { 3477c478bd9Sstevel@tonic-gate /* Close the track but leave session open */ 3487c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[2] = 1; 3497c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[5] = trackno & 0xff; 3507c478bd9Sstevel@tonic-gate } 3517c478bd9Sstevel@tonic-gate 3527c478bd9Sstevel@tonic-gate /* 3537c478bd9Sstevel@tonic-gate * DVD+R media are already formatted, we are using 3547c478bd9Sstevel@tonic-gate * a special case to notify that drive to close 3557c478bd9Sstevel@tonic-gate * track/session and null-fill the remaining space. 3567c478bd9Sstevel@tonic-gate */ 3577c478bd9Sstevel@tonic-gate if (device_type == DVD_PLUS) { 3587c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[5] = 1; /* only 1 track */ 3597c478bd9Sstevel@tonic-gate 3607c478bd9Sstevel@tonic-gate if (close_session) { 3617c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[2] = 6; /* session */ 3627c478bd9Sstevel@tonic-gate } else { 3637c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[2] = 1; /* track */ 3647c478bd9Sstevel@tonic-gate } 3657c478bd9Sstevel@tonic-gate } 3667c478bd9Sstevel@tonic-gate 3677c478bd9Sstevel@tonic-gate scmd->uscsi_cdblen = 10; 3687c478bd9Sstevel@tonic-gate if ((uscsi_error = uscsi(fd, scmd)) < 0) 3697c478bd9Sstevel@tonic-gate return (0); 3707c478bd9Sstevel@tonic-gate return (1); 3717c478bd9Sstevel@tonic-gate } 3727c478bd9Sstevel@tonic-gate 3737c478bd9Sstevel@tonic-gate int 3747c478bd9Sstevel@tonic-gate blank_disc(int fd, int type, int immediate) 3757c478bd9Sstevel@tonic-gate { 3767c478bd9Sstevel@tonic-gate struct uscsi_cmd *scmd; 3777c478bd9Sstevel@tonic-gate 3787c478bd9Sstevel@tonic-gate scmd = get_uscsi_cmd(); 3797c478bd9Sstevel@tonic-gate scmd->uscsi_flags = USCSI_SILENT; 3807c478bd9Sstevel@tonic-gate 3817c478bd9Sstevel@tonic-gate if (immediate) { 3827c478bd9Sstevel@tonic-gate scmd->uscsi_timeout = DEFAULT_SCSI_TIMEOUT; 3837c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[1] = 0x10; 3847c478bd9Sstevel@tonic-gate } else { 3857c478bd9Sstevel@tonic-gate scmd->uscsi_timeout = 0x12c0; 3867c478bd9Sstevel@tonic-gate } 3877c478bd9Sstevel@tonic-gate ((uchar_t *)scmd->uscsi_cdb)[0] = BLANK_CMD; 3887c478bd9Sstevel@tonic-gate 3897c478bd9Sstevel@tonic-gate /* tell it to blank the last session or all of the disk */ 3907c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[1] |= type & 0x07; 3917c478bd9Sstevel@tonic-gate scmd->uscsi_cdblen = 12; 3927c478bd9Sstevel@tonic-gate 3937c478bd9Sstevel@tonic-gate if ((uscsi_error = uscsi(fd, scmd)) < 0) 3947c478bd9Sstevel@tonic-gate return (0); 3957c478bd9Sstevel@tonic-gate return (1); 3967c478bd9Sstevel@tonic-gate } 3977c478bd9Sstevel@tonic-gate 3987c478bd9Sstevel@tonic-gate int 3997c478bd9Sstevel@tonic-gate read_cd(int fd, uint32_t start_blk, uint16_t nblk, uchar_t sector_type, 4007c478bd9Sstevel@tonic-gate uchar_t *buf, uint32_t bufsize) 4017c478bd9Sstevel@tonic-gate { 4027c478bd9Sstevel@tonic-gate struct uscsi_cmd *scmd; 4037c478bd9Sstevel@tonic-gate 4047c478bd9Sstevel@tonic-gate scmd = get_uscsi_cmd(); 4057c478bd9Sstevel@tonic-gate scmd->uscsi_flags = USCSI_READ|USCSI_SILENT; 4067c478bd9Sstevel@tonic-gate scmd->uscsi_timeout = DEFAULT_SCSI_TIMEOUT; 4077c478bd9Sstevel@tonic-gate ((uchar_t *)scmd->uscsi_cdb)[0] = READ_CD_CMD; 4087c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[1] = (sector_type & 0x7) << 2; 4097c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[5] = start_blk & 0xff; 4107c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[4] = (start_blk >> 8) & 0xff; 4117c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[3] = (start_blk >> 16) & 0xff; 4127c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[2] = (start_blk >> 24) & 0xff; 4137c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[8] = nblk & 0xff; 4147c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[7] = (nblk >> 8) & 0xff; 4157c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[9] = 0x10; 4167c478bd9Sstevel@tonic-gate scmd->uscsi_cdblen = 12; 4177c478bd9Sstevel@tonic-gate scmd->uscsi_bufaddr = (char *)buf; 4187c478bd9Sstevel@tonic-gate scmd->uscsi_buflen = bufsize; 4197c478bd9Sstevel@tonic-gate if ((uscsi_error = uscsi(fd, scmd)) < 0) 4207c478bd9Sstevel@tonic-gate return (0); 4217c478bd9Sstevel@tonic-gate return (1); 4227c478bd9Sstevel@tonic-gate } 4237c478bd9Sstevel@tonic-gate 4247c478bd9Sstevel@tonic-gate int 4257c478bd9Sstevel@tonic-gate load_unload(int fd, int load) 4267c478bd9Sstevel@tonic-gate { 4277c478bd9Sstevel@tonic-gate struct uscsi_cmd *scmd; 4287c478bd9Sstevel@tonic-gate 4297c478bd9Sstevel@tonic-gate scmd = get_uscsi_cmd(); 4307c478bd9Sstevel@tonic-gate scmd->uscsi_flags = USCSI_SILENT; 4317c478bd9Sstevel@tonic-gate scmd->uscsi_timeout = DEFAULT_SCSI_TIMEOUT; 4327c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[0] = START_STOP_CMD; 4337c478bd9Sstevel@tonic-gate if (load == 0) { 4347c478bd9Sstevel@tonic-gate /* unload medium */ 4357c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[4] = 2; 4367c478bd9Sstevel@tonic-gate } else { 4377c478bd9Sstevel@tonic-gate /* load medium */ 4387c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[4] = 3; 4397c478bd9Sstevel@tonic-gate } 4407c478bd9Sstevel@tonic-gate scmd->uscsi_cdblen = 6; 4417c478bd9Sstevel@tonic-gate 4427c478bd9Sstevel@tonic-gate if ((uscsi_error = uscsi(fd, scmd)) < 0) 4437c478bd9Sstevel@tonic-gate return (0); 4447c478bd9Sstevel@tonic-gate return (1); 4457c478bd9Sstevel@tonic-gate } 4467c478bd9Sstevel@tonic-gate 4477c478bd9Sstevel@tonic-gate int 4487c478bd9Sstevel@tonic-gate prevent_allow_mr(int fd, int op) 4497c478bd9Sstevel@tonic-gate { 4507c478bd9Sstevel@tonic-gate struct uscsi_cmd *scmd; 4517c478bd9Sstevel@tonic-gate 4527c478bd9Sstevel@tonic-gate scmd = get_uscsi_cmd(); 4537c478bd9Sstevel@tonic-gate scmd->uscsi_flags = USCSI_SILENT; 4547c478bd9Sstevel@tonic-gate scmd->uscsi_timeout = DEFAULT_SCSI_TIMEOUT; 4557c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[0] = PREVENT_ALLOW_CMD; 4567c478bd9Sstevel@tonic-gate if (!op) { /* prevent */ 4577c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[4] = 1; 4587c478bd9Sstevel@tonic-gate } 4597c478bd9Sstevel@tonic-gate scmd->uscsi_cdblen = 6; 4607c478bd9Sstevel@tonic-gate if ((uscsi_error = uscsi(fd, scmd)) < 0) 4617c478bd9Sstevel@tonic-gate return (0); 4627c478bd9Sstevel@tonic-gate return (1); 4637c478bd9Sstevel@tonic-gate } 4647c478bd9Sstevel@tonic-gate 4657c478bd9Sstevel@tonic-gate int 4667c478bd9Sstevel@tonic-gate set_cd_speed(int fd, uint16_t read_speed, uint16_t write_speed) 4677c478bd9Sstevel@tonic-gate { 4687c478bd9Sstevel@tonic-gate struct uscsi_cmd *scmd; 4697c478bd9Sstevel@tonic-gate 4707c478bd9Sstevel@tonic-gate scmd = get_uscsi_cmd(); 4717c478bd9Sstevel@tonic-gate scmd->uscsi_flags = USCSI_SILENT; 4727c478bd9Sstevel@tonic-gate scmd->uscsi_timeout = DEFAULT_SCSI_TIMEOUT; 4737c478bd9Sstevel@tonic-gate scmd->uscsi_cdblen = 0xc; 4747c478bd9Sstevel@tonic-gate ((uchar_t *)scmd->uscsi_cdb)[0] = SET_CD_SPEED; 4757c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[2] = (read_speed >> 8) & 0xff; 4767c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[3] = read_speed & 0xff; 4777c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[4] = (write_speed >> 8) & 0xff; 4787c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[5] = write_speed & 0xff; 4797c478bd9Sstevel@tonic-gate 4807c478bd9Sstevel@tonic-gate if ((uscsi_error = uscsi(fd, scmd)) < 0) 4817c478bd9Sstevel@tonic-gate return (0); 4827c478bd9Sstevel@tonic-gate return (1); 4837c478bd9Sstevel@tonic-gate } 4847c478bd9Sstevel@tonic-gate 4857c478bd9Sstevel@tonic-gate int 4867c478bd9Sstevel@tonic-gate get_performance(int fd, int get_write_performance, uchar_t *perf) 4877c478bd9Sstevel@tonic-gate { 4887c478bd9Sstevel@tonic-gate struct uscsi_cmd *scmd; 4897c478bd9Sstevel@tonic-gate 4907c478bd9Sstevel@tonic-gate scmd = get_uscsi_cmd(); 4917c478bd9Sstevel@tonic-gate scmd->uscsi_flags = USCSI_READ|USCSI_SILENT; 4927c478bd9Sstevel@tonic-gate scmd->uscsi_buflen = GET_PERF_DATA_LEN; 4937c478bd9Sstevel@tonic-gate scmd->uscsi_bufaddr = (char *)perf; 4947c478bd9Sstevel@tonic-gate scmd->uscsi_timeout = DEFAULT_SCSI_TIMEOUT; 4957c478bd9Sstevel@tonic-gate scmd->uscsi_cdblen = 0xc; 4967c478bd9Sstevel@tonic-gate ((uchar_t *)scmd->uscsi_cdb)[0] = GET_PERFORMANCE_CMD; 4977c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[1] = 0x10; 4987c478bd9Sstevel@tonic-gate if (get_write_performance) 4997c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[1] |= 4; 5007c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[9] = 2; 5017c478bd9Sstevel@tonic-gate if ((uscsi_error = uscsi(fd, scmd)) < 0) 5027c478bd9Sstevel@tonic-gate return (0); 5037c478bd9Sstevel@tonic-gate return (1); 5047c478bd9Sstevel@tonic-gate } 5057c478bd9Sstevel@tonic-gate 5067c478bd9Sstevel@tonic-gate int 5077c478bd9Sstevel@tonic-gate set_streaming(int fd, uchar_t *buf) 5087c478bd9Sstevel@tonic-gate { 5097c478bd9Sstevel@tonic-gate struct uscsi_cmd *scmd; 5107c478bd9Sstevel@tonic-gate 5117c478bd9Sstevel@tonic-gate scmd = get_uscsi_cmd(); 5127c478bd9Sstevel@tonic-gate scmd->uscsi_flags = USCSI_WRITE|USCSI_SILENT; 5137c478bd9Sstevel@tonic-gate scmd->uscsi_buflen = SET_STREAM_DATA_LEN; 5147c478bd9Sstevel@tonic-gate scmd->uscsi_bufaddr = (char *)buf; 5157c478bd9Sstevel@tonic-gate scmd->uscsi_timeout = DEFAULT_SCSI_TIMEOUT; 5167c478bd9Sstevel@tonic-gate scmd->uscsi_cdblen = 0xc; 5177c478bd9Sstevel@tonic-gate ((uchar_t *)scmd->uscsi_cdb)[0] = STREAM_CMD; 5187c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[10] = SET_STREAM_DATA_LEN; 5197c478bd9Sstevel@tonic-gate if ((uscsi_error = uscsi(fd, scmd)) < 0) 5207c478bd9Sstevel@tonic-gate return (0); 5217c478bd9Sstevel@tonic-gate return (1); 5227c478bd9Sstevel@tonic-gate } 5237c478bd9Sstevel@tonic-gate 5247c478bd9Sstevel@tonic-gate int 5257c478bd9Sstevel@tonic-gate rezero_unit(int fd) 5267c478bd9Sstevel@tonic-gate { 5277c478bd9Sstevel@tonic-gate struct uscsi_cmd *scmd; 5287c478bd9Sstevel@tonic-gate 5297c478bd9Sstevel@tonic-gate scmd = get_uscsi_cmd(); 5307c478bd9Sstevel@tonic-gate scmd->uscsi_flags = USCSI_SILENT; 5317c478bd9Sstevel@tonic-gate scmd->uscsi_timeout = DEFAULT_SCSI_TIMEOUT; 5327c478bd9Sstevel@tonic-gate scmd->uscsi_cdblen = 0x6; 5337c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[0] = REZERO_UNIT_CMD; 5347c478bd9Sstevel@tonic-gate if ((uscsi_error = uscsi(fd, scmd)) < 0) 5357c478bd9Sstevel@tonic-gate return (0); 5367c478bd9Sstevel@tonic-gate return (1); 5377c478bd9Sstevel@tonic-gate } 5387c478bd9Sstevel@tonic-gate 5397c478bd9Sstevel@tonic-gate int 5407c478bd9Sstevel@tonic-gate start_stop(int fd, int start) 5417c478bd9Sstevel@tonic-gate { 5427c478bd9Sstevel@tonic-gate struct uscsi_cmd *scmd; 5437c478bd9Sstevel@tonic-gate 5447c478bd9Sstevel@tonic-gate scmd = get_uscsi_cmd(); 5457c478bd9Sstevel@tonic-gate scmd->uscsi_flags = USCSI_SILENT; 5467c478bd9Sstevel@tonic-gate scmd->uscsi_timeout = DEFAULT_SCSI_TIMEOUT; 5477c478bd9Sstevel@tonic-gate scmd->uscsi_cdblen = 0x6; 5487c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[0] = START_STOP_CMD; 5497c478bd9Sstevel@tonic-gate if (start) { 5507c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[4] = 1; 5517c478bd9Sstevel@tonic-gate } 5527c478bd9Sstevel@tonic-gate if ((uscsi_error = uscsi(fd, scmd)) < 0) 5537c478bd9Sstevel@tonic-gate return (0); 5547c478bd9Sstevel@tonic-gate return (1); 5557c478bd9Sstevel@tonic-gate } 5567c478bd9Sstevel@tonic-gate 5577c478bd9Sstevel@tonic-gate int 5587c478bd9Sstevel@tonic-gate flush_cache(int fd) 5597c478bd9Sstevel@tonic-gate { 5607c478bd9Sstevel@tonic-gate struct uscsi_cmd *scmd; 5617c478bd9Sstevel@tonic-gate 5627c478bd9Sstevel@tonic-gate scmd = get_uscsi_cmd(); 5637c478bd9Sstevel@tonic-gate scmd->uscsi_flags = USCSI_SILENT; 5647c478bd9Sstevel@tonic-gate scmd->uscsi_timeout = DEFAULT_SCSI_TIMEOUT; 5657c478bd9Sstevel@tonic-gate scmd->uscsi_cdblen = 10; 5667c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[0] = SYNC_CACHE_CMD; 5677c478bd9Sstevel@tonic-gate if (device_type != CD_RW) { 5687c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[1] = 0x2; /* Immediate */ 5697c478bd9Sstevel@tonic-gate } 5707c478bd9Sstevel@tonic-gate 5717c478bd9Sstevel@tonic-gate if ((uscsi_error = uscsi(fd, scmd)) < 0) 5727c478bd9Sstevel@tonic-gate return (0); 5737c478bd9Sstevel@tonic-gate return (1); 5747c478bd9Sstevel@tonic-gate } 5757c478bd9Sstevel@tonic-gate 5767c478bd9Sstevel@tonic-gate /* 5777c478bd9Sstevel@tonic-gate * used for DVD- to reserve the size we want to write. 5787c478bd9Sstevel@tonic-gate * This is used by the drive to generate a TOC. 5797c478bd9Sstevel@tonic-gate */ 5807c478bd9Sstevel@tonic-gate int 5817c478bd9Sstevel@tonic-gate set_reservation(int fd, ulong_t size) 5827c478bd9Sstevel@tonic-gate { 5837c478bd9Sstevel@tonic-gate struct uscsi_cmd *scmd; 5847c478bd9Sstevel@tonic-gate 5857c478bd9Sstevel@tonic-gate scmd = get_uscsi_cmd(); 5867c478bd9Sstevel@tonic-gate scmd->uscsi_flags = USCSI_READ|USCSI_SILENT; 5877c478bd9Sstevel@tonic-gate scmd->uscsi_timeout = DEFAULT_SCSI_TIMEOUT; 5887c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[0] = SET_RESERVATION_CMD; 5897c478bd9Sstevel@tonic-gate scmd->uscsi_cdblen = 10; 5907c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[5] = (uchar_t)(size >> 24); 5917c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[6] = (uchar_t)(size >> 16); 5927c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[7] = (uchar_t)(size >> 8); 5937c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[8] = (uchar_t)size; 5947c478bd9Sstevel@tonic-gate if ((uscsi_error = uscsi(fd, scmd)) < 0) 5957c478bd9Sstevel@tonic-gate return (0); 5967c478bd9Sstevel@tonic-gate return (1); 5977c478bd9Sstevel@tonic-gate } 5987c478bd9Sstevel@tonic-gate 5997c478bd9Sstevel@tonic-gate /* 6007c478bd9Sstevel@tonic-gate * Used for DVD+RW media to prepare the disk to write. 6017c478bd9Sstevel@tonic-gate * It will also be used for packet mode writing when 6027c478bd9Sstevel@tonic-gate * it becomes supported. 6037c478bd9Sstevel@tonic-gate */ 6047c478bd9Sstevel@tonic-gate int 6057c478bd9Sstevel@tonic-gate format_media(int fd) 6067c478bd9Sstevel@tonic-gate { 6077c478bd9Sstevel@tonic-gate struct uscsi_cmd *scmd; 6087c478bd9Sstevel@tonic-gate uchar_t buf[20]; 6097c478bd9Sstevel@tonic-gate 6107c478bd9Sstevel@tonic-gate (void) memset(buf, 0, 20); 6117c478bd9Sstevel@tonic-gate scmd = get_uscsi_cmd(); 6127c478bd9Sstevel@tonic-gate scmd->uscsi_flags = USCSI_READ|USCSI_SILENT; 6137c478bd9Sstevel@tonic-gate scmd->uscsi_timeout = DEFAULT_SCSI_TIMEOUT; 6147c478bd9Sstevel@tonic-gate 6157c478bd9Sstevel@tonic-gate scmd->uscsi_cdblen = 12; 6167c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[0] = READ_FORMAT_CAP_CMD; 6177c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[8] = 0x14; /* buffer length */ 6187c478bd9Sstevel@tonic-gate scmd->uscsi_buflen = 20; 6197c478bd9Sstevel@tonic-gate scmd->uscsi_bufaddr = (char *)buf; 6207c478bd9Sstevel@tonic-gate 6217c478bd9Sstevel@tonic-gate if ((uscsi_error = uscsi(fd, scmd)) < 0) 6227c478bd9Sstevel@tonic-gate return (0); 6237c478bd9Sstevel@tonic-gate 6247c478bd9Sstevel@tonic-gate /* RE-use cap structure */ 6257c478bd9Sstevel@tonic-gate 6267c478bd9Sstevel@tonic-gate scmd->uscsi_flags = USCSI_WRITE|USCSI_SILENT; 6277c478bd9Sstevel@tonic-gate scmd->uscsi_timeout = DEFAULT_SCSI_TIMEOUT; 6287c478bd9Sstevel@tonic-gate scmd->uscsi_cdblen = 6; 6297c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[0] = FORMAT_UNIT_CMD; 6307c478bd9Sstevel@tonic-gate /* full format */ 6317c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[1] = 0x11; 6327c478bd9Sstevel@tonic-gate scmd->uscsi_buflen = 12; 6337c478bd9Sstevel@tonic-gate buf[1] = 0x82; /* immediate and FOV */ 6347c478bd9Sstevel@tonic-gate buf[3] = 8; /* descriptor length */ 6357c478bd9Sstevel@tonic-gate buf[8] = 0x98; /* type = 26 DVD+RW format */ 6367c478bd9Sstevel@tonic-gate buf[10] = 0; 6377c478bd9Sstevel@tonic-gate 6387c478bd9Sstevel@tonic-gate if ((uscsi_error = uscsi(fd, scmd)) < 0) 6397c478bd9Sstevel@tonic-gate return (0); 6407c478bd9Sstevel@tonic-gate return (1); 6417c478bd9Sstevel@tonic-gate } 6427c478bd9Sstevel@tonic-gate 6437c478bd9Sstevel@tonic-gate 6447c478bd9Sstevel@tonic-gate /* 6457c478bd9Sstevel@tonic-gate * Prefered method of reading the media size. This is 6467c478bd9Sstevel@tonic-gate * the only supported method on several newer drives. 6477c478bd9Sstevel@tonic-gate */ 6487c478bd9Sstevel@tonic-gate uint32_t 6497c478bd9Sstevel@tonic-gate read_format_capacity(int fd, uint_t *bsize) 6507c478bd9Sstevel@tonic-gate { 6517c478bd9Sstevel@tonic-gate struct uscsi_cmd *scmd; 6527c478bd9Sstevel@tonic-gate uint32_t filesize; 6537c478bd9Sstevel@tonic-gate char buf[20]; 6547c478bd9Sstevel@tonic-gate 6557c478bd9Sstevel@tonic-gate scmd = get_uscsi_cmd(); 6567c478bd9Sstevel@tonic-gate scmd->uscsi_flags = USCSI_READ|USCSI_SILENT; 6577c478bd9Sstevel@tonic-gate scmd->uscsi_timeout = DEFAULT_SCSI_TIMEOUT; 6587c478bd9Sstevel@tonic-gate scmd->uscsi_cdblen = 12; 6597c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[0] = READ_FORMAT_CAP_CMD; 6607c478bd9Sstevel@tonic-gate scmd->uscsi_cdb[8] = 0x14; 6617c478bd9Sstevel@tonic-gate scmd->uscsi_buflen = 20; 6627c478bd9Sstevel@tonic-gate scmd->uscsi_bufaddr = buf; 6637c478bd9Sstevel@tonic-gate 6647c478bd9Sstevel@tonic-gate if ((uscsi_error = uscsi(fd, scmd)) < 0) 6657c478bd9Sstevel@tonic-gate return (0); 6667c478bd9Sstevel@tonic-gate 6677c478bd9Sstevel@tonic-gate filesize = (uint32_t)(((uchar_t)buf[4] << 24) + 6687c478bd9Sstevel@tonic-gate ((uchar_t)buf[5] << 16) + ((uchar_t)buf[6] << 8) + (uchar_t)buf[7]); 6697c478bd9Sstevel@tonic-gate 6707c478bd9Sstevel@tonic-gate *bsize = (uint16_t)(((uchar_t)buf[10] << 8) + (uchar_t)buf[11]); 6717c478bd9Sstevel@tonic-gate 6727c478bd9Sstevel@tonic-gate return (filesize); 6737c478bd9Sstevel@tonic-gate } 674