/*************************************************************************** scsi.c - description ------------------- begin : Sun Nov 16 2003 copyright : (C) 2003 by Pete Bernert email : BlackDove@addcom.de ***************************************************************************/ /*************************************************************************** * * * This program is free software; you can redistribute it and/or modify * * it under the terms of the GNU General Public License as published by * * the Free Software Foundation; either version 2 of the License, or * * (at your option) any later version. See also the license.txt file for * * additional informations. * * * ***************************************************************************/ //*************************************************************************// // History of changes: // // 2003/11/16 - Pete // - generic cleanup for the Peops release // //*************************************************************************// ///////////////////////////////////////////////////////// #include "stdafx.h" #define _IN_SCSI #include "externals.h" ///////////////////////////////////////////////////////// SRB_ExecSCSICmd sx; // used with all (non-waiting) read funcs BOOL bDoWaiting=FALSE; // flag for async reads ///////////////////////////////////////////////////////// // returns device type int GetSCSIDevice(int iA,int iT,int iL) { SRB_GDEVBlock s;DWORD dwStatus; memset(&s,0,sizeof(SRB_GDEVBlock)); s.SRB_Cmd = SC_GET_DEV_TYPE; s.SRB_HaID = iA; s.SRB_Target = iT; s.SRB_Lun = iL; ResetEvent(hEvent); dwStatus=pSendASPI32Command((LPSRB)&s); if(dwStatus==SS_PENDING) {WaitGenEvent(30000);dwStatus=s.SRB_Status;} if(dwStatus==SS_COMP) return s.SRB_DeviceType; return -1; } ///////////////////////////////////////////////////////// int GetSCSIStatus(int iA,int iT,int iL) { SRB_ExecSCSICmd s;DWORD dwStatus; char ret[0x324]; memset(&s,0,sizeof(s)); s.SRB_Cmd = SC_EXEC_SCSI_CMD; s.SRB_HaId = iCD_AD; s.SRB_Target = iCD_TA; s.SRB_Lun = iCD_LU; s.SRB_Flags = SRB_DIR_IN | SRB_EVENT_NOTIFY; s.SRB_BufLen = 0x324; s.SRB_BufPointer = (BYTE FAR *)ret; s.SRB_SenseLen = 0x0E; s.SRB_CDBLen = 0x0A; s.SRB_PostProc = (LPVOID)hEvent; s.CDBByte[0] = 0x00; ResetEvent(hEvent); dwStatus=pSendASPI32Command((LPSRB)&s); if(dwStatus==SS_PENDING) WaitGenEvent(30000); return s.SRB_Status; } ///////////////////////////////////////////////////////// // fills toc infos DWORD GetSCSITOC(LPTOC toc) { SRB_ExecSCSICmd s;DWORD dwStatus; memset(&s,0,sizeof(s)); s.SRB_Cmd = SC_EXEC_SCSI_CMD; s.SRB_HaId = iCD_AD; s.SRB_Target = iCD_TA; s.SRB_Lun = iCD_LU; s.SRB_Flags = SRB_DIR_IN | SRB_EVENT_NOTIFY; s.SRB_BufLen = 0x324; s.SRB_BufPointer = (BYTE FAR *)toc; s.SRB_SenseLen = 0x0E; s.SRB_CDBLen = 0x0A; s.SRB_PostProc = (LPVOID)hEvent; s.CDBByte[0] = 0x43; s.CDBByte[1] = 0x00; // 0x02 for MSF s.CDBByte[7] = 0x03; s.CDBByte[8] = 0x24; ResetEvent(hEvent); dwStatus=pSendASPI32Command((LPSRB)&s); if(dwStatus==SS_PENDING) WaitGenEvent(30000); if(s.SRB_Status!=SS_COMP) return SS_ERR; return SS_COMP; } ///////////////////////////////////////////////////////// // enum all cd drives into 32k buffer, return num of drives int GetSCSICDDrives(char * pDList) { int iCnt=0,iA,iT,iL;char * p=pDList; SRB_HAInquiry si;SRB_GDEVBlock sd; SRB_ExecSCSICmd s;int iNumA;char szBuf[100]; DWORD dw,dwStatus; if(!pGetASPI32SupportInfo) return 0; dw=pGetASPI32SupportInfo(); if(HIBYTE(LOWORD(dw))!=SS_COMP) return 0; iNumA=(int)LOBYTE(LOWORD(dw)); for(iA=0;iA<iNumA;iA++) { memset(&si,0,sizeof(SRB_HAInquiry)); si.SRB_Cmd = SC_HA_INQUIRY; si.SRB_HaId = iA; pSendASPI32Command((LPSRB)&si); if(si.SRB_Status!=SS_COMP) continue; if(!si.HA_Unique[3]) si.HA_Unique[3]=8; for(iT=0;iT<si.HA_Unique[3];iT++) { for(iL=0;iL<8;iL++) { memset(&sd,0,sizeof(SRB_GDEVBlock)); sd.SRB_Cmd = SC_GET_DEV_TYPE; sd.SRB_HaID = iA; sd.SRB_Target = iT; sd.SRB_Lun = iL; pSendASPI32Command((LPSRB)&sd); if(sd.SRB_Status==SS_COMP && sd.SRB_DeviceType==DTYPE_CDROM) { memset(&s,0,sizeof(s)); s.SRB_Cmd = SC_EXEC_SCSI_CMD; s.SRB_HaId = iA; s.SRB_Target = iT; s.SRB_Lun = iL; s.SRB_Flags = SRB_DIR_IN | SRB_EVENT_NOTIFY; s.SRB_BufLen = 100; s.SRB_BufPointer = szBuf; s.SRB_SenseLen = SENSE_LEN; s.SRB_CDBLen = 6; s.SRB_PostProc = (LPVOID)hEvent; s.CDBByte[0] = SCSI_INQUIRY; s.CDBByte[4] = 100; ResetEvent(hEvent); dwStatus=pSendASPI32Command((LPSRB)&s); if(dwStatus==SS_PENDING) WaitGenEvent(WAITFOREVER); if(s.SRB_Status==SS_COMP) { int i; for(i=8 ;i<15;i++) if(szBuf[i]==' ') {szBuf[i]=0;break;} szBuf[15]=0; for(i=16;i<32;i++) if(szBuf[i]==' ') {szBuf[i]=0;break;} szBuf[31]=0; for(i=32;i<37;i++) if(szBuf[i]==' ') {szBuf[i]=0;break;} szBuf[36]=0; wsprintf(p,"[%d:%d:%d] %s %s V%s", iA,iT,iL,&szBuf[8],&szBuf[16],&szBuf[32]); iCnt++; p+=strlen(p)+1; } } } } } return iCnt; } ///////////////////////////////////////////////////////// // play audio DWORD PlaySCSIAudio(unsigned long start,unsigned long len) { SRB_ExecSCSICmd s;DWORD dwStatus; memset(&s,0,sizeof(s)); s.SRB_Cmd = SC_EXEC_SCSI_CMD; s.SRB_HaId = iCD_AD; s.SRB_Target = iCD_TA; s.SRB_Lun = iCD_LU; s.SRB_Flags = SRB_DIR_IN | SRB_EVENT_NOTIFY; s.SRB_BufLen = 0; s.SRB_BufPointer = 0; s.SRB_SenseLen = SENSE_LEN; s.SRB_CDBLen = 12; s.SRB_PostProc = (LPVOID)hEvent; if(start==0) // mmm... that stop doesn't seem to work on all drives... pfff { s.CDBByte[0] = 0x4b; s.CDBByte[1] = (iCD_LU << 5) | 01; s.CDBByte[2] = 0; s.CDBByte[3] = 0; s.CDBByte[4] = 0; s.CDBByte[5] = 0; s.CDBByte[6] = 0; s.CDBByte[7] = 0; s.CDBByte[8] = 0; s.CDBByte[9] = 0; } else // start playing { s.CDBByte[0] = 0xa5; s.CDBByte[1] = iCD_LU << 5; s.CDBByte[2] = (unsigned char)((start >> 24) & 0xFF); s.CDBByte[3] = (unsigned char)((start >> 16) & 0xFF); s.CDBByte[4] = (unsigned char)((start >> 8) & 0xFF); s.CDBByte[5] = (unsigned char)((start & 0xFF)); s.CDBByte[6] = (unsigned char)((len >> 24) & 0xFF); s.CDBByte[7] = (unsigned char)((len >> 16) & 0xFF); s.CDBByte[8] = (unsigned char)((len >> 8) & 0xFF); s.CDBByte[9] = (unsigned char)(len & 0xFF); } ResetEvent(hEvent); dwStatus = pSendASPI32Command((LPSRB)&s); if(dwStatus==SS_PENDING) WaitGenEvent(10000); return s.SRB_Status; } ///////////////////////////////////////////////////////// // do (unprecise) sub channel read on audio play unsigned char * GetSCSIAudioSub(void) { SRB_ExecSCSICmd s;DWORD dwStatus; unsigned char cB[20]; memset(cB,0,20); memset(&s,0,sizeof(s)); s.SRB_Cmd = SC_EXEC_SCSI_CMD; s.SRB_HaId = iCD_AD; s.SRB_Target = iCD_TA; s.SRB_Lun = iCD_LU; s.SRB_Flags = SRB_DIR_IN | SRB_EVENT_NOTIFY; s.SRB_SenseLen = SENSE_LEN; s.SRB_BufLen = 20;//44; s.SRB_BufPointer = cB; s.SRB_CDBLen = 10; s.CDBByte[0] = 0x42; s.CDBByte[1] = (iCD_LU<<5)|2; // lun & msf s.CDBByte[2] = 0x40; // subq s.CDBByte[3] = 0x01; // curr pos info s.CDBByte[6] = 0; // track number (only in isrc mode, ignored) s.CDBByte[7] = 0; // alloc len s.CDBByte[8] = 20;//44; ResetEvent(hEvent); dwStatus = pSendASPI32Command((LPSRB)&s); if(dwStatus==SS_PENDING) WaitGenEvent(WAITFOREVER); if(s.SRB_Status!=SS_COMP) return NULL; SubAData[12]=(cB[5]<<4)|(cB[5]>>4); SubAData[13]=cB[6]; SubAData[14]=cB[7]; SubAData[15]=itob(cB[13]); SubAData[16]=itob(cB[14]); SubAData[17]=itob(cB[15]); SubAData[18]=0; SubAData[19]=itob(cB[9]); SubAData[20]=itob(cB[10]); SubAData[21]=itob(cB[11]); return SubAData; } ///////////////////////////////////////////////////////// // test, if drive is ready (doesn't work on all drives) BOOL TestSCSIUnitReady(void) { SRB_ExecSCSICmd s;DWORD dwStatus; memset(&s,0,sizeof(s)); s.SRB_Cmd = SC_EXEC_SCSI_CMD; s.SRB_HaId = iCD_AD; s.SRB_Target = iCD_TA; s.SRB_Lun = iCD_LU; s.SRB_Flags = SRB_DIR_IN | SRB_EVENT_NOTIFY; s.SRB_BufLen = 0; s.SRB_BufPointer = 0; s.SRB_SenseLen = SENSE_LEN; s.SRB_CDBLen = 6; s.SRB_PostProc = (LPVOID)hEvent; s.CDBByte[0] = 0x00; s.CDBByte[1] = iCD_LU << 5; ResetEvent(hEvent); dwStatus = pSendASPI32Command((LPSRB)&s); if(dwStatus==SS_PENDING) WaitGenEvent(1000); if(s.SRB_Status!=SS_COMP) return FALSE; if(s.SRB_TargStat==STATUS_GOOD) return TRUE; // will always be GOOD with ioctl, so no problem here return FALSE; } ///////////////////////////////////////////////////////// // change the read speed (not supported on all drives) DWORD SetSCSISpeed(DWORD dwSpeed) { SRB_ExecSCSICmd s;DWORD dwStatus; memset(&s,0,sizeof(s)); s.SRB_Cmd = SC_EXEC_SCSI_CMD; s.SRB_HaId = iCD_AD; s.SRB_Target = iCD_TA; s.SRB_Lun = iCD_LU; s.SRB_Flags = SRB_DIR_OUT | SRB_EVENT_NOTIFY; s.SRB_SenseLen = SENSE_LEN; s.SRB_CDBLen = 12; s.SRB_PostProc = (LPVOID)hEvent; s.CDBByte[0] = 0xBB; s.CDBByte[2] = (BYTE)(dwSpeed >> 8); s.CDBByte[3] = (BYTE)dwSpeed; ResetEvent(hEvent); dwStatus=pSendASPI32Command((LPSRB)&s); if(dwStatus==SS_PENDING) WaitGenEvent(WAITFOREVER); if(s.SRB_Status!=SS_COMP) return SS_ERR; return SS_COMP; } ///////////////////////////////////////////////////////// ///////////////////////////////////////////////////////// // all the different SCSI read commands can be found here // 'bWait' is a flag, if the command should wait until // completed, or if the func can return as soon as possible // (async reading). Attention: 'bWait' is not really used // in the Sub-channel commands yet (sub is done always // blocking, and just 'one sector' reads are done, caching=0) ///////////////////////////////////////////////////////// ///////////////////////////////////////////////////////// // BE: used by most ATAPI drives ///////////////////////////////////////////////////////// DWORD ReadSCSI_BE(BOOL bWait,FRAMEBUF * f) { DWORD dwStatus; memset(&sx,0,sizeof(sx)); sx.SRB_Cmd = SC_EXEC_SCSI_CMD; sx.SRB_HaId = iCD_AD; sx.SRB_Target = iCD_TA; sx.SRB_Lun = iCD_LU; sx.SRB_Flags = SRB_DIR_IN | SRB_EVENT_NOTIFY; sx.SRB_BufLen = f->dwBufLen; sx.SRB_BufPointer = &(f->BufData[0]); sx.SRB_SenseLen = SENSE_LEN; sx.SRB_CDBLen = 12; sx.SRB_PostProc = (LPVOID)hEvent; sx.CDBByte[0] = 0xBE; //s.CDBByte[1] = 0x04; sx.CDBByte[2] = (unsigned char)((f->dwFrame >> 24) & 0xFF); sx.CDBByte[3] = (unsigned char)((f->dwFrame >> 16) & 0xFF); sx.CDBByte[4] = (unsigned char)((f->dwFrame >> 8) & 0xFF); sx.CDBByte[5] = (unsigned char)(f->dwFrame & 0xFF); sx.CDBByte[8] = (unsigned char)(f->dwFrameCnt & 0xFF); sx.CDBByte[9] = (iRType==MODE_BE_1)?0x10:0xF8;//F0!!!!!!!!!!! ResetEvent(hEvent); dwStatus=pSendASPI32Command((LPSRB)&sx); if(dwStatus==SS_PENDING) { if(bWait) WaitGenEvent(WAITFOREVER); else { bDoWaiting=TRUE; return SS_COMP; } } if(sx.SRB_Status!=SS_COMP) return SS_ERR; return SS_COMP; } ///////////////////////////////////////////////////////// DWORD ReadSCSI_BE_Sub(BOOL bWait,FRAMEBUF * f) { DWORD dwStatus; memset(&sx,0,sizeof(sx)); sx.SRB_Cmd = SC_EXEC_SCSI_CMD; sx.SRB_HaId = iCD_AD; sx.SRB_Target = iCD_TA; sx.SRB_Lun = iCD_LU; sx.SRB_Flags = SRB_DIR_IN | SRB_EVENT_NOTIFY; sx.SRB_BufLen = f->dwBufLen + 16; sx.SRB_BufPointer = &(f->BufData[0]); sx.SRB_SenseLen = SENSE_LEN; sx.SRB_CDBLen = 12; sx.SRB_PostProc = (LPVOID)hEvent; sx.CDBByte[0] = 0xBE; //s.CDBByte[1] = 0x04; sx.CDBByte[2] = (unsigned char)((f->dwFrame >> 24) & 0xFF); sx.CDBByte[3] = (unsigned char)((f->dwFrame >> 16) & 0xFF); sx.CDBByte[4] = (unsigned char)((f->dwFrame >> 8) & 0xFF); sx.CDBByte[5] = (unsigned char)(f->dwFrame & 0xFF); sx.CDBByte[8] = (unsigned char)(f->dwFrameCnt & 0xFF); sx.CDBByte[9] = (iRType==MODE_BE_1)?0x10:0xF8;//F0!!!!!!!!!!! sx.CDBByte[10] = 0x2; ResetEvent(hEvent ); dwStatus=pSendASPI32Command((LPSRB)&sx); if(dwStatus==SS_PENDING) WaitGenEvent(WAITSUB); if(sx.SRB_Status!=SS_COMP) return SS_ERR; memcpy(&SubCData[12],&f->BufData[2352],16); SubCData[15]=itob(SubCData[15]); SubCData[16]=itob(SubCData[16]); SubCData[17]=itob(SubCData[17]); SubCData[19]=itob(SubCData[19]); SubCData[20]=itob(SubCData[20]); SubCData[21]=itob(SubCData[21]); return SS_COMP; } ///////////////////////////////////////////////////////// // different sub reading for lite-on ltd163d... // 16 bytes subc data is encoded in 96 bytes... DWORD ReadSCSI_BE_Sub_1(BOOL bWait,FRAMEBUF * f) { DWORD dwStatus; memset(&sx,0,sizeof(sx)); sx.SRB_Cmd = SC_EXEC_SCSI_CMD; sx.SRB_HaId = iCD_AD; sx.SRB_Target = iCD_TA; sx.SRB_Lun = iCD_LU; sx.SRB_Flags = SRB_DIR_IN | SRB_EVENT_NOTIFY; sx.SRB_BufLen = f->dwBufLen + 96; sx.SRB_BufPointer = &(f->BufData[0]); sx.SRB_SenseLen = SENSE_LEN; sx.SRB_CDBLen = 12; sx.SRB_PostProc = (LPVOID)hEvent; sx.CDBByte[0] = 0xBE; sx.CDBByte[2] = (unsigned char)((f->dwFrame >> 24) & 0xFF); sx.CDBByte[3] = (unsigned char)((f->dwFrame >> 16) & 0xFF); sx.CDBByte[4] = (unsigned char)((f->dwFrame >> 8) & 0xFF); sx.CDBByte[5] = (unsigned char)(f->dwFrame & 0xFF); sx.CDBByte[8] = (unsigned char)(f->dwFrameCnt & 0xFF); sx.CDBByte[9] = 0xF8; sx.CDBByte[10] = 0x1; ResetEvent(hEvent); dwStatus=pSendASPI32Command((LPSRB)&sx); if(dwStatus==SS_PENDING) WaitGenEvent(WAITSUB); if(sx.SRB_Status!=SS_COMP) return SS_ERR; DecodeSub_BE_2_1(&f->BufData[2352]); memcpy(&SubCData[12],&f->BufData[2352],16); return SS_COMP; } ///////////////////////////////////////////////////////// // 28: used by most SCSI drives ///////////////////////////////////////////////////////// DWORD InitSCSI_28_2(void) { SRB_ExecSCSICmd s;DWORD dwStatus; int i; BYTE init1[] = { 0, 0, 0, 0x08, 0, 0, 0, 0, 0, 0, 0x09, 0x30, 0x23, 6, 0, 0, 0, 0, 0, 0x80 }; BYTE init2[] = { 0, 0, 0, 8, 0, 0, 0, 0, 0, 0, 9, 48, 1, 6, 32, 7, 0, 0, 0, 0 }; for(i=0;i<2;i++) { memset( &s, 0, sizeof( s ) ); s.SRB_Cmd = SC_EXEC_SCSI_CMD; s.SRB_HaId = iCD_AD; s.SRB_Target = iCD_TA; s.SRB_Lun = iCD_LU; s.SRB_Flags = SRB_EVENT_NOTIFY; s.SRB_BufLen = 0x14; s.SRB_BufPointer = (i==0)?init1:init2; s.SRB_SenseLen = SENSE_LEN; s.SRB_CDBLen = 6; s.SRB_PostProc = (LPVOID)hEvent; s.CDBByte[0] = 0x15; s.CDBByte[1] = 0x10; s.CDBByte[4] = 0x14; ResetEvent(hEvent); dwStatus=pSendASPI32Command((LPSRB)&s); if (dwStatus == SS_PENDING) WaitGenEvent(WAITFOREVER); if(s.SRB_Status!=SS_COMP) return SS_ERR; } pDeInitFunc = DeInitSCSI_28; return s.SRB_Status; } ///////////////////////////////////////////////////////// DWORD InitSCSI_28_1(void) { SRB_ExecSCSICmd s;DWORD dwStatus; BYTE init1[] = { 0, 0, 0, 0x08, 0, 0, 0, 0, 0, 0, 0x09, 0x30 }; memset(&s,0,sizeof(s)); s.SRB_Cmd = SC_EXEC_SCSI_CMD; s.SRB_HaId = iCD_AD; s.SRB_Target = iCD_TA; s.SRB_Lun = iCD_LU; s.SRB_Flags = SRB_EVENT_NOTIFY; s.SRB_BufLen = 0x0C; s.SRB_BufPointer = init1; s.SRB_SenseLen = SENSE_LEN; s.SRB_CDBLen = 6; s.SRB_PostProc = (LPVOID)hEvent; s.CDBByte[0] = 0x15; s.CDBByte[4] = 0x0C; ResetEvent(hEvent); dwStatus=pSendASPI32Command((LPSRB)&s); if(dwStatus==SS_PENDING) WaitGenEvent(WAITFOREVER); if(s.SRB_Status!=SS_COMP) return SS_ERR; pDeInitFunc = DeInitSCSI_28; return s.SRB_Status; } ///////////////////////////////////////////////////////// DWORD InitSCSI_28_2048(void) { SRB_ExecSCSICmd s;DWORD dwStatus; BYTE init1[] = { 0, 0, 0, 0x08, 0, 0, 0, 0, 0, 0, 0x08, 0x0 }; pDeInitFunc = DeInitSCSI_28; memset(&s,0,sizeof(s)); s.SRB_Cmd = SC_EXEC_SCSI_CMD; s.SRB_HaId = iCD_AD; s.SRB_Target = iCD_TA; s.SRB_Lun = iCD_LU; s.SRB_Flags = SRB_EVENT_NOTIFY; s.SRB_BufLen = 0x0C; s.SRB_BufPointer = init1; s.SRB_SenseLen = SENSE_LEN; s.SRB_CDBLen = 6; s.SRB_PostProc = (LPVOID)hEvent; s.CDBByte[0] = 0x15; s.CDBByte[4] = 0x0C; ResetEvent(hEvent); dwStatus=pSendASPI32Command((LPSRB)&s); if(dwStatus==SS_PENDING) WaitGenEvent(WAITFOREVER); if(s.SRB_Status!=SS_COMP) return SS_ERR; return s.SRB_Status; } ///////////////////////////////////////////////////////// DWORD DeInitSCSI_28(void) { SRB_ExecSCSICmd s;DWORD dwStatus; BYTE init1[] = { 0, 0, 0, 8, 83, 0, 0, 0, 0, 0, 8, 0 }; memset(&s,0,sizeof(s)); s.SRB_Cmd = SC_EXEC_SCSI_CMD; s.SRB_HaId = iCD_AD; s.SRB_Target = iCD_TA; s.SRB_Lun = iCD_LU; s.SRB_Flags = SRB_EVENT_NOTIFY | SRB_ENABLE_RESIDUAL_COUNT; s.SRB_BufLen = 0x0C; s.SRB_BufPointer = init1; s.SRB_SenseLen = SENSE_LEN; s.SRB_CDBLen = 6; s.SRB_PostProc = (LPVOID)hEvent; s.CDBByte[0] = 0x15; s.CDBByte[4] = 0x0C; ResetEvent(hEvent); dwStatus=pSendASPI32Command((LPSRB)&s); if(dwStatus==SS_PENDING) WaitGenEvent(WAITFOREVER); if(s.SRB_Status!=SS_COMP) return SS_ERR; return s.SRB_Status; } ///////////////////////////////////////////////////////// DWORD ReadSCSI_28(BOOL bWait,FRAMEBUF * f) { DWORD dwStatus; if(!pDeInitFunc) { if(iRType==MODE_28_2) { if(InitSCSI_28_2()!=SS_COMP) return SS_ERR; } else { if(InitSCSI_28_1()!=SS_COMP) return SS_ERR; } } memset(&sx,0,sizeof(sx)); sx.SRB_Cmd = SC_EXEC_SCSI_CMD; sx.SRB_HaId = iCD_AD; sx.SRB_Target = iCD_TA; sx.SRB_Lun = iCD_LU; sx.SRB_Flags = SRB_DIR_IN | SRB_EVENT_NOTIFY; sx.SRB_BufLen = f->dwBufLen; sx.SRB_BufPointer = &(f->BufData[0]); sx.SRB_SenseLen = SENSE_LEN; sx.SRB_CDBLen = 10; sx.SRB_PostProc = (LPVOID)hEvent; sx.CDBByte[0] = 0x28; // read10 command sx.CDBByte[1] = iCD_LU << 5; sx.CDBByte[2] = (unsigned char)((f->dwFrame >> 24) & 0xFF); sx.CDBByte[3] = (unsigned char)((f->dwFrame >> 16) & 0xFF); sx.CDBByte[4] = (unsigned char)((f->dwFrame >> 8) & 0xFF); sx.CDBByte[5] = (unsigned char)(f->dwFrame & 0xFF); sx.CDBByte[8] = (unsigned char)(f->dwFrameCnt & 0xFF); ResetEvent(hEvent); dwStatus=pSendASPI32Command((LPSRB)&sx); if(dwStatus==SS_PENDING) { if(bWait) WaitGenEvent(WAITFOREVER); else { bDoWaiting=TRUE; return SS_COMP; } } if(sx.SRB_Status!=SS_COMP) return SS_ERR; return SS_COMP; } ///////////////////////////////////////////////////////// // DVD MODE DWORD ReadSCSI_28_2048(BOOL bWait,FRAMEBUF * f) { DWORD dwStatus; if(!pDeInitFunc) { InitSCSI_28_2048(); //if(InitSCSI_28_2048()!=SS_COMP) return SS_ERR; } memset(&sx,0,sizeof(sx)); sx.SRB_Cmd = SC_EXEC_SCSI_CMD; sx.SRB_HaId = iCD_AD; sx.SRB_Target = iCD_TA; sx.SRB_Lun = iCD_LU; sx.SRB_Flags = SRB_DIR_IN | SRB_EVENT_NOTIFY; sx.SRB_BufLen = f->dwBufLen; sx.SRB_BufPointer = &(f->BufData[0]); sx.SRB_SenseLen = SENSE_LEN; sx.SRB_CDBLen = 10; sx.SRB_PostProc = (LPVOID)hEvent; sx.CDBByte[0] = 0x28; // read10 command sx.CDBByte[1] = iCD_LU << 5; sx.CDBByte[2] = (unsigned char)((f->dwFrame >> 24) & 0xFF); sx.CDBByte[3] = (unsigned char)((f->dwFrame >> 16) & 0xFF); sx.CDBByte[4] = (unsigned char)((f->dwFrame >> 8) & 0xFF); sx.CDBByte[5] = (unsigned char)(f->dwFrame & 0xFF); sx.CDBByte[8] = (unsigned char)(f->dwFrameCnt & 0xFF); sx.CDBByte[9] = 0xF8; ResetEvent(hEvent); dwStatus=pSendASPI32Command((LPSRB)&sx); if(dwStatus==SS_PENDING) { if(bWait) WaitGenEvent(WAITFOREVER); else { bDoWaiting=TRUE; return SS_COMP; } } if(sx.SRB_Status!=SS_COMP) return SS_ERR; return SS_COMP; } DWORD ReadSCSI_28_2048_Ex(BOOL bWait,FRAMEBUF * f) { DWORD dwStatus; if(!pDeInitFunc) { InitSCSI_28_2048(); //if(InitSCSI_28_2048()!=SS_COMP) return SS_ERR; } memset(&sx,0,sizeof(sx)); sx.SRB_Cmd = SC_EXEC_SCSI_CMD; sx.SRB_HaId = iCD_AD; sx.SRB_Target = iCD_TA; sx.SRB_Lun = iCD_LU; sx.SRB_Flags = SRB_DIR_IN | SRB_EVENT_NOTIFY; sx.SRB_BufLen = f->dwBufLen; sx.SRB_BufPointer = &(f->BufData[0]); sx.SRB_SenseLen = SENSE_LEN; sx.SRB_CDBLen = 10; sx.SRB_PostProc = (LPVOID)hEvent; sx.CDBByte[0] = 0x28; // read10 command sx.CDBByte[1] = iCD_LU << 5; sx.CDBByte[2] = (unsigned char)((f->dwFrame >> 24) & 0xFF); sx.CDBByte[3] = (unsigned char)((f->dwFrame >> 16) & 0xFF); sx.CDBByte[4] = (unsigned char)((f->dwFrame >> 8) & 0xFF); sx.CDBByte[5] = (unsigned char)(f->dwFrame & 0xFF); sx.CDBByte[8] = (unsigned char)(f->dwFrameCnt & 0xFF); // NO F8 //sx.CDBByte[9] = 0xF8; ResetEvent(hEvent); dwStatus=pSendASPI32Command((LPSRB)&sx); if(dwStatus==SS_PENDING) { if(bWait) WaitGenEvent(WAITFOREVER); else { bDoWaiting=TRUE; return SS_COMP; } } if(sx.SRB_Status!=SS_COMP) return SS_ERR; return SS_COMP; } ///////////////////////////////////////////////////////// // stupid subc reading on Teac 532S char tbuf[2368]; DWORD ReadSCSI_28_Sub(BOOL bWait,FRAMEBUF * f) { DWORD dwStatus; if(!pDeInitFunc) { if(iRType==MODE_28_2) { if(InitSCSI_28_2()!=SS_COMP) return SS_ERR; } else { if(InitSCSI_28_1()!=SS_COMP) return SS_ERR; } } memset(&sx,0,sizeof(sx)); sx.SRB_Cmd = SC_EXEC_SCSI_CMD; sx.SRB_HaId = iCD_AD; sx.SRB_Target = iCD_TA; sx.SRB_Lun = iCD_LU; sx.SRB_Flags = SRB_DIR_IN | SRB_EVENT_NOTIFY; sx.SRB_BufLen = f->dwBufLen; sx.SRB_BufPointer = &(f->BufData[0]); sx.SRB_SenseLen = SENSE_LEN; sx.SRB_CDBLen = 10; sx.SRB_PostProc = (LPVOID)hEvent; sx.CDBByte[0] = 0x28; // read10 sx.CDBByte[1] = iCD_LU << 5; sx.CDBByte[2] = (unsigned char)((f->dwFrame >> 24) & 0xFF); sx.CDBByte[3] = (unsigned char)((f->dwFrame >> 16) & 0xFF); sx.CDBByte[4] = (unsigned char)((f->dwFrame >> 8) & 0xFF); sx.CDBByte[5] = (unsigned char)(f->dwFrame & 0xFF); sx.CDBByte[8] = (unsigned char)(f->dwFrameCnt & 0xFF); ResetEvent(hEvent); dwStatus=pSendASPI32Command((LPSRB)&sx); if(dwStatus==SS_PENDING) WaitGenEvent(WAITFOREVER); if(sx.SRB_Status!=SS_COMP) return SS_ERR; memset(&sx,0,sizeof(sx)); sx.SRB_Cmd = SC_EXEC_SCSI_CMD; sx.SRB_HaId = iCD_AD; sx.SRB_Target = iCD_TA; sx.SRB_Lun = iCD_LU; sx.SRB_Flags = SRB_DIR_IN | SRB_EVENT_NOTIFY; sx.SRB_BufLen = 2368; sx.SRB_BufPointer = tbuf; sx.SRB_SenseLen = SENSE_LEN; sx.SRB_CDBLen = 12; sx.SRB_PostProc = (LPVOID)hEvent; sx.CDBByte[0] = 0xD8; sx.CDBByte[2] = (unsigned char)((f->dwFrame >> 24) & 0xFF); sx.CDBByte[3] = (unsigned char)((f->dwFrame >> 16) & 0xFF); sx.CDBByte[4] = (unsigned char)((f->dwFrame >> 8) & 0xFF); sx.CDBByte[5] = (unsigned char)(f->dwFrame & 0xFF); sx.CDBByte[9] = (unsigned char)(f->dwFrameCnt & 0xFF); sx.CDBByte[10] = 1; ResetEvent(hEvent); dwStatus=pSendASPI32Command((LPSRB)&sx); if(dwStatus==SS_PENDING) WaitGenEvent(WAITFOREVER); if(sx.SRB_Status!=SS_COMP) return SS_ERR; memcpy(&SubCData[12],&tbuf[2352],16); return SS_COMP; } ///////////////////////////////////////////////////////// // various simple scsi sub data read funcs... used for // ripping and subread checking... first 2352 bytes can // be trash after read, only the bytes after are important ///////////////////////////////////////////////////////// ///////////////////////////////////////////////////////// int ReadSub_BE_2(unsigned long addr,unsigned char * pBuf,int iNum) { DWORD dwStatus; memset(&sx,0,sizeof(sx)); sx.SRB_Cmd = SC_EXEC_SCSI_CMD; sx.SRB_HaId = iCD_AD; sx.SRB_Target = iCD_TA; sx.SRB_Lun = iCD_LU; sx.SRB_Flags = SRB_DIR_IN | SRB_EVENT_NOTIFY; sx.SRB_BufLen = 2368*iNum; sx.SRB_BufPointer = pBuf; sx.SRB_SenseLen = SENSE_LEN; sx.SRB_CDBLen = 12; sx.SRB_PostProc = (LPVOID)hEvent; sx.CDBByte[0] = 0xBE; sx.CDBByte[2] = (unsigned char)((addr >> 24) & 0xFF); sx.CDBByte[3] = (unsigned char)((addr >> 16) & 0xFF); sx.CDBByte[4] = (unsigned char)((addr >> 8) & 0xFF); sx.CDBByte[5] = (unsigned char)(addr & 0xFF); sx.CDBByte[8] = iNum; sx.CDBByte[9] = 0xF8; sx.CDBByte[10] = 0x2; ResetEvent(hEvent); dwStatus=pSendASPI32Command( (LPSRB)&sx ); if(dwStatus==SS_PENDING) WaitGenEvent(WAITSUB); if(sx.SRB_Status!=SS_COMP) return 0; return 1; } ///////////////////////////////////////////////////////// int ReadSub_BE_2_1(unsigned long addr,unsigned char * pBuf,int iNum) { DWORD dwStatus; memset(&sx,0,sizeof(sx)); sx.SRB_Cmd = SC_EXEC_SCSI_CMD; sx.SRB_HaId = iCD_AD; sx.SRB_Target = iCD_TA; sx.SRB_Lun = iCD_LU; sx.SRB_Flags = SRB_DIR_IN | SRB_EVENT_NOTIFY; sx.SRB_BufLen = 2448*iNum; // special! 96 bytes instead of 16 sx.SRB_BufPointer = pBuf; sx.SRB_SenseLen = SENSE_LEN; sx.SRB_CDBLen = 12; sx.SRB_PostProc = (LPVOID)hEvent; sx.CDBByte[0] = 0xBE; sx.CDBByte[2] = (unsigned char)((addr >> 24) & 0xFF); sx.CDBByte[3] = (unsigned char)((addr >> 16) & 0xFF); sx.CDBByte[4] = (unsigned char)((addr >> 8) & 0xFF); sx.CDBByte[5] = (unsigned char)(addr & 0xFF); sx.CDBByte[8] = iNum; sx.CDBByte[9] = 0xF8; sx.CDBByte[10] = 0x1; ResetEvent(hEvent); dwStatus=pSendASPI32Command((LPSRB)&sx); if(dwStatus==SS_PENDING) WaitGenEvent(WAITSUB); if(sx.SRB_Status!=SS_COMP) return 0; return 1; } ///////////////////////////////////////////////////////// int ReadSub_D8(unsigned long addr,unsigned char * pBuf,int iNum) { DWORD dwStatus; memset(&sx,0,sizeof(sx)); sx.SRB_Cmd = SC_EXEC_SCSI_CMD; sx.SRB_HaId = iCD_AD; sx.SRB_Target = iCD_TA; sx.SRB_Lun = iCD_LU; sx.SRB_Flags = SRB_DIR_IN | SRB_EVENT_NOTIFY; sx.SRB_BufLen = 2368*iNum; sx.SRB_BufPointer = pBuf; sx.SRB_SenseLen = SENSE_LEN; sx.SRB_CDBLen = 12; sx.SRB_PostProc = (LPVOID)hEvent; sx.CDBByte[0] = 0xD8; sx.CDBByte[2] = (unsigned char)((addr >> 24) & 0xFF); sx.CDBByte[3] = (unsigned char)((addr >> 16) & 0xFF); sx.CDBByte[4] = (unsigned char)((addr >> 8) & 0xFF); sx.CDBByte[5] = (unsigned char)(addr & 0xFF); sx.CDBByte[9] = iNum; sx.CDBByte[10] = 1; ResetEvent(hEvent); dwStatus=pSendASPI32Command((LPSRB)&sx); if(dwStatus==SS_PENDING) WaitGenEvent(WAITFOREVER); if(sx.SRB_Status!=SS_COMP) return 0; return 1; } ///////////////////////////////////////////////////////// // liteon subdata decoding void DecodeSub_BE_2_1(unsigned char * pBuf) { int i,j; unsigned char * pS=pBuf; unsigned char c; for(i=0;i<12;i++) { c=0; for(j=7;j>=0;j--,pS++) { if(*pS & 0x40) c|=(1<<j); } *(pBuf+i)=c; } for(i=12;i<16;i++) *(pBuf+i)=0; } ///////////////////////////////////////////////////////// // auto-detection of read mode (without subc), called // on plugin startup #define MAXMODES 6 DWORD CheckSCSIReadMode(void) { DWORD dwStatus;int i,j,k,iCnt;char * p; int iModes[MAXMODES]={MODE_BE_2,MODE_BE_1,MODE_28_1,MODE_28_2,MODE_28_2048,MODE_28_2048_Ex}; int iBlock[MAXMODES]={2352, 2352, 2352, 2352, 2048, 2048}; unsigned char cdb[3000]; FRAMEBUF * f=(FRAMEBUF *)cdb; for(i=0;i<MAXMODES;i++) // loop avail read modes { f->dwFrame = 16; // we check on addr 16 (should be available on all ps2 cds/dvds) f->dwFrameCnt = 1; f->dwBufLen = iBlock[i]; pDeInitFunc = NULL; iRType=iModes[i]; // set global read mode GetGenReadFunc(iRType); // get read func pointer for(j=0;j<3;j++) // try it 3 times { memset(f->BufData,0xAA,f->dwBufLen); // fill buf with AA dwStatus=pReadFunc(TRUE,f); // do the read #ifdef DBGOUT auxprintf("status %d\n",dwStatus); #endif if(dwStatus!=SS_COMP) continue; // error? try again p=&(f->BufData[0]); #ifdef DBGOUT auxprintf("check mode %d\n",i); #endif for(k=0,iCnt=0;k<(int)f->dwBufLen;k+=4,p+=4) // now check the returned data { #ifdef DBGOUT // auxprintf("%08x ",*((DWORD *)p)); #endif if(*((DWORD *)p)==0xAAAAAAAA) // -> still AA? bad iCnt++; else iCnt=0; if(iCnt>=8) {dwStatus=SS_ERR;break;} // -> if we have found many AA's, the reading was bad } if(dwStatus==SS_COMP) // reading was a success, no AA's? { iRType = iModes[i]; // -> set found mode iUsedBlockSize = iBlock[i]; if(iUsedBlockSize==2352) iUsedMode=CDVD_MODE_2352; else iUsedMode=CDVD_MODE_2048; #ifdef DBGOUT auxprintf("mode found %d\n",i); #endif return dwStatus; // -> bye } } if(pDeInitFunc) pDeInitFunc(); // deinit, try next mode } return dwStatus; } ///////////////////////////////////////////////////////// // dummy read dunc DWORD ReadSCSI_Dummy(BOOL bWait,FRAMEBUF * f) { return SS_ERR; } /////////////////////////////////////////////////////////