Patrik Stridvall | 9633632 | 1999-10-24 22:13:47 +0000 | [diff] [blame^] | 1 | #include "config.h" |
| 2 | |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 3 | #include <stdlib.h> |
| 4 | #include <sys/types.h> |
| 5 | #include <sys/stat.h> |
| 6 | #include <errno.h> |
| 7 | #include <fcntl.h> |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 8 | #include <memory.h> |
| 9 | #include <unistd.h> |
Marcus Meissner | 317af32 | 1999-02-17 13:51:06 +0000 | [diff] [blame] | 10 | |
| 11 | #include "winbase.h" |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 12 | #include "aspi.h" |
| 13 | #include "wnaspi32.h" |
| 14 | #include "options.h" |
| 15 | #include "heap.h" |
Alexandre Julliard | a099a55 | 1999-06-12 15:45:58 +0000 | [diff] [blame] | 16 | #include "debugtools.h" |
Patrik Stridvall | 021bd85 | 1999-07-18 18:40:11 +0000 | [diff] [blame] | 17 | #include "ldt.h" |
| 18 | #include "callback.h" |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 19 | |
Patrik Stridvall | b4b9fae | 1999-04-19 14:56:29 +0000 | [diff] [blame] | 20 | DEFAULT_DEBUG_CHANNEL(aspi) |
| 21 | |
Marcus Meissner | 7e09874 | 1999-10-13 12:14:26 +0000 | [diff] [blame] | 22 | #define malloc(x) HeapAlloc(GetProcessHeap(),0,x) |
| 23 | #define free(p) HeapFree(GetProcessHeap(),0,p) |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 24 | |
| 25 | /* FIXME! |
| 26 | * 1) Residual byte length reporting not handled |
| 27 | * 2) Make this code re-entrant for multithreading |
| 28 | * 3) Only linux supported so far |
| 29 | */ |
| 30 | |
| 31 | #ifdef linux |
Patrik Stridvall | 1ed4ecf | 1999-06-26 14:58:24 +0000 | [diff] [blame] | 32 | |
| 33 | static ASPI_DEVICE_INFO *ASPI_open_devices = NULL; |
| 34 | |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 35 | static int |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 36 | ASPI_OpenDevice(SRB_ExecSCSICmd *prb) |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 37 | { |
| 38 | int fd; |
| 39 | char idstr[20]; |
| 40 | char device_str[50]; |
| 41 | ASPI_DEVICE_INFO *curr; |
| 42 | |
| 43 | /* search list of devices to see if we've opened it already. |
| 44 | * There is not an explicit open/close in ASPI land, so hopefully |
| 45 | * keeping a device open won't be a problem. |
| 46 | */ |
| 47 | |
| 48 | for (curr = ASPI_open_devices; curr; curr = curr->next) { |
| 49 | if (curr->hostId == prb->SRB_HaId && |
| 50 | curr->target == prb->SRB_Target && |
| 51 | curr->lun == prb->SRB_Lun) { |
| 52 | return curr->fd; |
| 53 | } |
| 54 | } |
| 55 | |
| 56 | /* device wasn't cached, go ahead and open it */ |
| 57 | sprintf(idstr, "scsi c%1dt%1dd%1d", prb->SRB_HaId, prb->SRB_Target, prb->SRB_Lun); |
| 58 | |
| 59 | if (!PROFILE_GetWineIniString(idstr, "Device", "", device_str, sizeof(device_str))) { |
Alexandre Julliard | a099a55 | 1999-06-12 15:45:58 +0000 | [diff] [blame] | 60 | TRACE("Trying to open unlisted scsi device %s\n", idstr); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 61 | return -1; |
| 62 | } |
| 63 | |
Alexandre Julliard | a099a55 | 1999-06-12 15:45:58 +0000 | [diff] [blame] | 64 | TRACE("Opening device %s=%s\n", idstr, device_str); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 65 | |
| 66 | fd = open(device_str, O_RDWR); |
| 67 | if (fd == -1) { |
| 68 | int save_error = errno; |
Andreas Mohr | 869bc25 | 1998-12-18 17:34:58 +0000 | [diff] [blame] | 69 | #ifdef HAVE_STRERROR |
Alexandre Julliard | a099a55 | 1999-06-12 15:45:58 +0000 | [diff] [blame] | 70 | ERR("Error opening device %s, error '%s'\n", device_str, strerror(save_error)); |
Andreas Mohr | 869bc25 | 1998-12-18 17:34:58 +0000 | [diff] [blame] | 71 | #else |
Alexandre Julliard | a099a55 | 1999-06-12 15:45:58 +0000 | [diff] [blame] | 72 | ERR("Error opening device %s, error %d\n", device_str, save_error); |
Andreas Mohr | 869bc25 | 1998-12-18 17:34:58 +0000 | [diff] [blame] | 73 | #endif |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 74 | return -1; |
| 75 | } |
| 76 | |
| 77 | /* device is now open */ |
| 78 | curr = HeapAlloc( SystemHeap, 0, sizeof(ASPI_DEVICE_INFO) ); |
| 79 | curr->fd = fd; |
| 80 | curr->hostId = prb->SRB_HaId; |
| 81 | curr->target = prb->SRB_Target; |
| 82 | curr->lun = prb->SRB_Lun; |
| 83 | |
| 84 | /* insert new record at beginning of open device list */ |
| 85 | curr->next = ASPI_open_devices; |
| 86 | ASPI_open_devices = curr; |
| 87 | return fd; |
| 88 | } |
| 89 | |
| 90 | |
| 91 | static void |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 92 | ASPI_DebugPrintCmd(SRB_ExecSCSICmd *prb) |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 93 | { |
| 94 | BYTE cmd; |
| 95 | int i; |
| 96 | BYTE *cdb; |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 97 | |
| 98 | switch (prb->CDBByte[0]) { |
| 99 | case CMD_INQUIRY: |
Alexandre Julliard | a099a55 | 1999-06-12 15:45:58 +0000 | [diff] [blame] | 100 | TRACE("{\n"); |
| 101 | TRACE("\tEVPD: %d\n", prb->CDBByte[1] & 1); |
| 102 | TRACE("\tLUN: %d\n", (prb->CDBByte[1] & 0xc) >> 1); |
| 103 | TRACE("\tPAGE CODE: %d\n", prb->CDBByte[2]); |
| 104 | TRACE("\tALLOCATION LENGTH: %d\n", prb->CDBByte[4]); |
| 105 | TRACE("\tCONTROL: %d\n", prb->CDBByte[5]); |
| 106 | TRACE("}\n"); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 107 | break; |
| 108 | case CMD_SCAN_SCAN: |
Alexandre Julliard | a099a55 | 1999-06-12 15:45:58 +0000 | [diff] [blame] | 109 | TRACE("Transfer Length: %d\n", prb->CDBByte[4]); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 110 | break; |
| 111 | } |
| 112 | |
Alexandre Julliard | a099a55 | 1999-06-12 15:45:58 +0000 | [diff] [blame] | 113 | TRACE("Host Adapter: %d\n", prb->SRB_HaId); |
| 114 | TRACE("Flags: %d\n", prb->SRB_Flags); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 115 | if (TARGET_TO_HOST(prb)) { |
Alexandre Julliard | a099a55 | 1999-06-12 15:45:58 +0000 | [diff] [blame] | 116 | TRACE("\tData transfer: Target to host. Length checked.\n"); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 117 | } |
| 118 | else if (HOST_TO_TARGET(prb)) { |
Alexandre Julliard | a099a55 | 1999-06-12 15:45:58 +0000 | [diff] [blame] | 119 | TRACE("\tData transfer: Host to target. Length checked.\n"); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 120 | } |
| 121 | else if (NO_DATA_TRANSFERED(prb)) { |
Alexandre Julliard | a099a55 | 1999-06-12 15:45:58 +0000 | [diff] [blame] | 122 | TRACE("\tData transfer: none\n"); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 123 | } |
| 124 | else { |
Alexandre Julliard | a099a55 | 1999-06-12 15:45:58 +0000 | [diff] [blame] | 125 | WARN("\tTransfer by scsi cmd. Length not checked.\n"); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 126 | } |
| 127 | |
Alexandre Julliard | a099a55 | 1999-06-12 15:45:58 +0000 | [diff] [blame] | 128 | TRACE("\tResidual byte length reporting %s\n", prb->SRB_Flags & 0x4 ? "enabled" : "disabled"); |
| 129 | TRACE("\tLinking %s\n", prb->SRB_Flags & 0x2 ? "enabled" : "disabled"); |
| 130 | TRACE("\tPosting %s\n", prb->SRB_Flags & 0x1 ? "enabled" : "disabled"); |
| 131 | TRACE("Target: %d\n", prb->SRB_Target); |
| 132 | TRACE("Lun: %d\n", prb->SRB_Lun); |
| 133 | TRACE("BufLen: %ld\n", prb->SRB_BufLen); |
| 134 | TRACE("SenseLen: %d\n", prb->SRB_SenseLen); |
| 135 | TRACE("BufPtr: %p\n", prb->SRB_BufPointer); |
| 136 | TRACE("CDB Length: %d\n", prb->SRB_CDBLen); |
| 137 | TRACE("POST Proc: %lx\n", (DWORD) prb->SRB_PostProc); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 138 | cdb = &prb->CDBByte[0]; |
| 139 | cmd = prb->CDBByte[0]; |
Alexandre Julliard | 15de615 | 1999-08-04 12:22:42 +0000 | [diff] [blame] | 140 | if (TRACE_ON(aspi)) |
| 141 | { |
| 142 | DPRINTF("CDB buffer["); |
| 143 | for (i = 0; i < prb->SRB_CDBLen; i++) { |
| 144 | if (i != 0) DPRINTF(","); |
| 145 | DPRINTF("%02x", *cdb++); |
| 146 | } |
| 147 | DPRINTF("]\n"); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 148 | } |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 149 | } |
| 150 | |
| 151 | static void |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 152 | ASPI_PrintSenseArea(SRB_ExecSCSICmd *prb) |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 153 | { |
| 154 | int i; |
| 155 | BYTE *cdb; |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 156 | |
Alexandre Julliard | 15de615 | 1999-08-04 12:22:42 +0000 | [diff] [blame] | 157 | if (TRACE_ON(aspi)) |
| 158 | { |
| 159 | cdb = &prb->CDBByte[0]; |
| 160 | DPRINTF("SenseArea["); |
| 161 | for (i = 0; i < prb->SRB_SenseLen; i++) { |
| 162 | if (i) DPRINTF(","); |
| 163 | DPRINTF("%02x", *cdb++); |
| 164 | } |
| 165 | DPRINTF("]\n"); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 166 | } |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 167 | } |
| 168 | |
| 169 | static void |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 170 | ASPI_DebugPrintResult(SRB_ExecSCSICmd *prb) |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 171 | { |
| 172 | |
| 173 | switch (prb->CDBByte[0]) { |
| 174 | case CMD_INQUIRY: |
Alexandre Julliard | a099a55 | 1999-06-12 15:45:58 +0000 | [diff] [blame] | 175 | TRACE("Vendor: '%s'\n", prb->SRB_BufPointer + INQUIRY_VENDOR); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 176 | break; |
| 177 | case CMD_TEST_UNIT_READY: |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 178 | ASPI_PrintSenseArea(prb); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 179 | break; |
| 180 | } |
| 181 | } |
| 182 | |
| 183 | static WORD |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 184 | ASPI_ExecScsiCmd(SRB_ExecSCSICmd *lpPRB) |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 185 | { |
| 186 | struct sg_header *sg_hd, *sg_reply_hdr; |
| 187 | int status; |
| 188 | int in_len, out_len; |
| 189 | int error_code = 0; |
| 190 | int fd; |
| 191 | |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 192 | ASPI_DebugPrintCmd(lpPRB); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 193 | |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 194 | fd = ASPI_OpenDevice(lpPRB); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 195 | if (fd == -1) { |
Alexandre Julliard | a099a55 | 1999-06-12 15:45:58 +0000 | [diff] [blame] | 196 | ERR("Failed: could not open device c%01dt%01dd%01d. Device permissions !?\n", |
Uwe Bonnes | a9a6d00 | 1999-06-06 14:30:08 +0000 | [diff] [blame] | 197 | lpPRB->SRB_HaId,lpPRB->SRB_Target,lpPRB->SRB_Lun); |
| 198 | lpPRB->SRB_Status = SS_NO_DEVICE; |
| 199 | return SS_NO_DEVICE; |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 200 | } |
| 201 | |
| 202 | sg_hd = NULL; |
| 203 | sg_reply_hdr = NULL; |
| 204 | |
| 205 | lpPRB->SRB_Status = SS_PENDING; |
| 206 | |
| 207 | if (!lpPRB->SRB_CDBLen) { |
Alexandre Julliard | a099a55 | 1999-06-12 15:45:58 +0000 | [diff] [blame] | 208 | WARN("Failed: lpPRB->SRB_CDBLen = 0.\n"); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 209 | lpPRB->SRB_Status = SS_ERR; |
| 210 | return SS_ERR; |
| 211 | } |
| 212 | |
| 213 | /* build up sg_header + scsi cmd */ |
| 214 | if (HOST_TO_TARGET(lpPRB)) { |
| 215 | /* send header, command, and then data */ |
| 216 | in_len = SCSI_OFF + lpPRB->SRB_CDBLen + lpPRB->SRB_BufLen; |
| 217 | sg_hd = (struct sg_header *) malloc(in_len); |
| 218 | memset(sg_hd, 0, SCSI_OFF); |
| 219 | memcpy(sg_hd + 1, &lpPRB->CDBByte[0], lpPRB->SRB_CDBLen); |
| 220 | if (lpPRB->SRB_BufLen) { |
| 221 | memcpy(((BYTE *) sg_hd) + SCSI_OFF + lpPRB->SRB_CDBLen, lpPRB->SRB_BufPointer, lpPRB->SRB_BufLen); |
| 222 | } |
| 223 | } |
| 224 | else { |
| 225 | /* send header and command - no data */ |
| 226 | in_len = SCSI_OFF + lpPRB->SRB_CDBLen; |
| 227 | sg_hd = (struct sg_header *) malloc(in_len); |
| 228 | memset(sg_hd, 0, SCSI_OFF); |
| 229 | memcpy(sg_hd + 1, &lpPRB->CDBByte[0], lpPRB->SRB_CDBLen); |
| 230 | } |
| 231 | |
| 232 | if (TARGET_TO_HOST(lpPRB)) { |
| 233 | out_len = SCSI_OFF + lpPRB->SRB_BufLen; |
| 234 | sg_reply_hdr = (struct sg_header *) malloc(out_len); |
| 235 | memset(sg_reply_hdr, 0, SCSI_OFF); |
| 236 | sg_hd->reply_len = out_len; |
| 237 | } |
| 238 | else { |
| 239 | out_len = SCSI_OFF; |
| 240 | sg_reply_hdr = (struct sg_header *) malloc(out_len); |
| 241 | memset(sg_reply_hdr, 0, SCSI_OFF); |
| 242 | sg_hd->reply_len = out_len; |
| 243 | } |
| 244 | |
| 245 | status = write(fd, sg_hd, in_len); |
| 246 | if (status < 0 || status != in_len) { |
Andreas Mohr | 869bc25 | 1998-12-18 17:34:58 +0000 | [diff] [blame] | 247 | int save_error = errno; |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 248 | |
Alexandre Julliard | a099a55 | 1999-06-12 15:45:58 +0000 | [diff] [blame] | 249 | WARN("Not enough bytes written to scsi device bytes=%d .. %d\n", in_len, status); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 250 | if (status < 0) { |
Andreas Mohr | 869bc25 | 1998-12-18 17:34:58 +0000 | [diff] [blame] | 251 | if (save_error == ENOMEM) { |
Alexandre Julliard | a099a55 | 1999-06-12 15:45:58 +0000 | [diff] [blame] | 252 | MESSAGE("ASPI: Linux generic scsi driver\n You probably need to re-compile your kernel with a larger SG_BIG_BUFF value (sg.h)\n Suggest 130560\n"); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 253 | } |
Andreas Mohr | 869bc25 | 1998-12-18 17:34:58 +0000 | [diff] [blame] | 254 | #ifdef HAVE_STRERROR |
Alexandre Julliard | a099a55 | 1999-06-12 15:45:58 +0000 | [diff] [blame] | 255 | WARN("error:= '%s'\n", strerror(save_error)); |
Andreas Mohr | 869bc25 | 1998-12-18 17:34:58 +0000 | [diff] [blame] | 256 | #else |
Alexandre Julliard | a099a55 | 1999-06-12 15:45:58 +0000 | [diff] [blame] | 257 | WARN("error:= %d\n", save_error); |
Andreas Mohr | 869bc25 | 1998-12-18 17:34:58 +0000 | [diff] [blame] | 258 | #endif |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 259 | } |
| 260 | goto error_exit; |
| 261 | } |
| 262 | |
| 263 | status = read(fd, sg_reply_hdr, out_len); |
| 264 | if (status < 0 || status != out_len) { |
Alexandre Julliard | a099a55 | 1999-06-12 15:45:58 +0000 | [diff] [blame] | 265 | WARN("not enough bytes read from scsi device%d\n", status); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 266 | goto error_exit; |
| 267 | } |
| 268 | |
| 269 | if (sg_reply_hdr->result != 0) { |
| 270 | error_code = sg_reply_hdr->result; |
Alexandre Julliard | a099a55 | 1999-06-12 15:45:58 +0000 | [diff] [blame] | 271 | WARN("reply header error (%d)\n", sg_reply_hdr->result); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 272 | goto error_exit; |
| 273 | } |
| 274 | |
| 275 | if (TARGET_TO_HOST(lpPRB) && lpPRB->SRB_BufLen) { |
| 276 | memcpy(lpPRB->SRB_BufPointer, sg_reply_hdr + 1, lpPRB->SRB_BufLen); |
| 277 | } |
| 278 | |
| 279 | /* copy in sense buffer to amount that is available in client */ |
| 280 | if (lpPRB->SRB_SenseLen) { |
| 281 | int sense_len = lpPRB->SRB_SenseLen; |
| 282 | if (lpPRB->SRB_SenseLen > 16) |
| 283 | sense_len = 16; |
| 284 | memcpy(SENSE_BUFFER(lpPRB), &sg_reply_hdr->sense_buffer[0], sense_len); |
| 285 | } |
| 286 | |
| 287 | |
| 288 | lpPRB->SRB_Status = SS_COMP; |
| 289 | lpPRB->SRB_HaStat = HASTAT_OK; |
| 290 | lpPRB->SRB_TargStat = STATUS_GOOD; |
| 291 | |
| 292 | /* now do posting */ |
| 293 | |
| 294 | if (lpPRB->SRB_PostProc) { |
| 295 | if (ASPI_POSTING(lpPRB)) { |
Alexandre Julliard | a099a55 | 1999-06-12 15:45:58 +0000 | [diff] [blame] | 296 | TRACE("Post Routine (%lx) called\n", (DWORD) lpPRB->SRB_PostProc); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 297 | (*lpPRB->SRB_PostProc)(lpPRB); |
| 298 | } |
| 299 | else |
| 300 | if (lpPRB->SRB_Flags & SRB_EVENT_NOTIFY) { |
Alexandre Julliard | a099a55 | 1999-06-12 15:45:58 +0000 | [diff] [blame] | 301 | TRACE("Setting event %04x\n", (HANDLE)lpPRB->SRB_PostProc); |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 302 | SetEvent((HANDLE)lpPRB->SRB_PostProc); /* FIXME: correct ? */ |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 303 | } |
| 304 | } |
| 305 | free(sg_reply_hdr); |
| 306 | free(sg_hd); |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 307 | ASPI_DebugPrintResult(lpPRB); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 308 | return SS_COMP; |
| 309 | |
| 310 | error_exit: |
| 311 | if (error_code == EBUSY) { |
| 312 | lpPRB->SRB_Status = SS_ASPI_IS_BUSY; |
Alexandre Julliard | a099a55 | 1999-06-12 15:45:58 +0000 | [diff] [blame] | 313 | TRACE("Device busy\n"); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 314 | } |
| 315 | else { |
Alexandre Julliard | a099a55 | 1999-06-12 15:45:58 +0000 | [diff] [blame] | 316 | WARN("Failed\n"); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 317 | lpPRB->SRB_Status = SS_ERR; |
| 318 | } |
| 319 | |
| 320 | /* I'm not sure exactly error codes work here |
| 321 | * We probably should set lpPRB->SRB_TargStat, SRB_HaStat ? |
| 322 | */ |
Alexandre Julliard | a099a55 | 1999-06-12 15:45:58 +0000 | [diff] [blame] | 323 | WARN("error_exit\n"); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 324 | free(sg_reply_hdr); |
| 325 | free(sg_hd); |
| 326 | return lpPRB->SRB_Status; |
| 327 | } |
| 328 | #endif |
| 329 | |
| 330 | |
| 331 | /******************************************************************* |
| 332 | * GetASPI32SupportInfo32 [WNASPI32.0] |
| 333 | * |
| 334 | * Checks if the ASPI subsystem is initialized correctly. |
| 335 | * |
| 336 | * RETURNS |
| 337 | * HIWORD: 0. |
| 338 | * HIBYTE of LOWORD: status (SS_COMP or SS_FAILED_INIT) |
| 339 | * LOBYTE of LOWORD: # of host adapters. |
| 340 | */ |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 341 | DWORD WINAPI GetASPI32SupportInfo() |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 342 | { |
| 343 | return ((SS_COMP << 8) | 1); /* FIXME: get # of host adapters installed */ |
| 344 | } |
| 345 | |
| 346 | |
| 347 | /*********************************************************************** |
| 348 | * SendASPI32Command32 (WNASPI32.1) |
| 349 | */ |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 350 | DWORD __cdecl SendASPI32Command(LPSRB lpSRB) |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 351 | { |
| 352 | #ifdef linux |
| 353 | switch (lpSRB->common.SRB_Cmd) { |
| 354 | case SC_HA_INQUIRY: |
| 355 | lpSRB->inquiry.SRB_Status = SS_COMP; /* completed successfully */ |
| 356 | lpSRB->inquiry.HA_Count = 1; /* not always */ |
| 357 | lpSRB->inquiry.HA_SCSI_ID = 7; /* not always ID 7 */ |
| 358 | strcat(lpSRB->inquiry.HA_ManagerId, "ASPI for WIN32"); /* max 15 chars, don't change */ |
| 359 | strcat(lpSRB->inquiry.HA_Identifier, "Wine host"); /* FIXME: return host adapter name */ |
| 360 | memset(lpSRB->inquiry.HA_Unique, 0, 16); /* default HA_Unique content */ |
| 361 | lpSRB->inquiry.HA_Unique[6] = 0x02; /* Maximum Transfer Length (128K, Byte> 4-7) */ |
Alexandre Julliard | a099a55 | 1999-06-12 15:45:58 +0000 | [diff] [blame] | 362 | FIXME("ASPI: Partially implemented SC_HA_INQUIRY for adapter %d.\n", lpSRB->inquiry.SRB_HaId); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 363 | return SS_COMP; |
Marcus Meissner | eb71e50 | 1999-09-19 12:08:18 +0000 | [diff] [blame] | 364 | case SC_GET_DEV_TYPE: { |
| 365 | SRB tmpsrb; |
| 366 | char inqbuf[200]; |
| 367 | |
| 368 | memset(&tmpsrb,0,sizeof(tmpsrb)); |
| 369 | |
| 370 | tmpsrb.cmd.SRB_Cmd = SC_EXEC_SCSI_CMD; |
| 371 | #define X(x) tmpsrb.cmd.SRB_##x = lpSRB->devtype.SRB_##x |
| 372 | X(Status);X(HaId);X(Flags);X(Target);X(Lun); |
| 373 | #undef X |
| 374 | tmpsrb.cmd.SRB_BufLen = sizeof(inqbuf); |
| 375 | tmpsrb.cmd.SRB_BufPointer = inqbuf; |
| 376 | tmpsrb.cmd.CDBByte[0] = 0x12; /* INQUIRY */ |
| 377 | tmpsrb.cmd.CDBByte[4] = sizeof(inqbuf); |
| 378 | tmpsrb.cmd.SRB_CDBLen = 6; |
| 379 | ASPI_ExecScsiCmd(&tmpsrb.cmd); |
| 380 | #define X(x) lpSRB->devtype.SRB_##x = tmpsrb.cmd.SRB_##x |
| 381 | X(Status); |
| 382 | #undef X |
| 383 | lpSRB->devtype.SRB_DeviceType = inqbuf[0]>>3; |
| 384 | FIXME("returning devicetype %d for target %d\n",inqbuf[0]>>3,tmpsrb.cmd.SRB_Target); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 385 | break; |
Marcus Meissner | eb71e50 | 1999-09-19 12:08:18 +0000 | [diff] [blame] | 386 | } |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 387 | case SC_EXEC_SCSI_CMD: |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 388 | return ASPI_ExecScsiCmd(&lpSRB->cmd); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 389 | break; |
| 390 | case SC_RESET_DEV: |
Alexandre Julliard | a099a55 | 1999-06-12 15:45:58 +0000 | [diff] [blame] | 391 | FIXME("Not implemented SC_RESET_DEV\n"); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 392 | break; |
| 393 | default: |
Alexandre Julliard | a099a55 | 1999-06-12 15:45:58 +0000 | [diff] [blame] | 394 | WARN("Unknown command %d\n", lpSRB->common.SRB_Cmd); |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 395 | } |
| 396 | return SS_INVALID_SRB; |
| 397 | #else |
| 398 | return SS_INVALID_SRB; |
| 399 | #endif |
| 400 | } |
| 401 | |
| 402 | |
| 403 | /*********************************************************************** |
| 404 | * GetASPI32DLLVersion32 (WNASPI32.3) |
| 405 | */ |
| 406 | |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 407 | DWORD WINAPI GetASPI32DLLVersion() |
Alexandre Julliard | d30dfd2 | 1998-09-27 18:28:36 +0000 | [diff] [blame] | 408 | { |
| 409 | #ifdef linux |
| 410 | return (DWORD)1; |
| 411 | #else |
| 412 | return (DWORD)0; |
| 413 | #endif |
| 414 | } |
| 415 | |