Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

857 lines
22 KiB

  1. //+-------------------------------------------------------------------------
  2. //
  3. // Microsoft Windows
  4. //
  5. // Copyright (C) Microsoft Corporation, 1998 - 1999
  6. //
  7. // File: wave.cpp
  8. //
  9. //--------------------------------------------------------------------------
  10. #define DIRECTSOUND_VERSION 0x600
  11. #include "stdafx.h"
  12. #include "Direct.h"
  13. #include "dSound.h"
  14. #include "dms.h"
  15. #include <mmreg.h>
  16. #include <msacm.h>
  17. // FOURCC codes
  18. #undef FOURCC_RIFF
  19. #define FOURCC_RIFF 'FFIR'
  20. #undef FOURCC_MEM
  21. #define FOURCC_MEM ' MEM'
  22. #undef FOURCC_WAVE
  23. #define FOURCC_WAVE 'EVAW'
  24. #undef FOURCC_FORMAT
  25. #define FOURCC_FORMAT ' tmf'
  26. #undef FOURCC_DATA
  27. #define FOURCC_DATA 'atad'
  28. #define RPF(level,str,err) \
  29. { char outBuf[MAX_PATH]; \
  30. wsprintf(outBuf,str,err); \
  31. OutputDebugString(outBuf); \
  32. }
  33. #define DPFLVL_ERROR 1
  34. /***************************************************************************
  35. *
  36. * FillWfx
  37. *
  38. * Description:
  39. * Fills a WAVEFORMATEX structure, given only the necessary values.
  40. *
  41. * Arguments:
  42. * LPWAVEFORMATEX [out]: structure to fill.
  43. * WORD [in]: number of channels.
  44. * DWORD [in]: samples per second.
  45. * WORD [in]: bits per sample.
  46. *
  47. * Returns:
  48. * (void)
  49. *
  50. ***************************************************************************/
  51. #undef DPF_FNAME
  52. void FillWfx(LPWAVEFORMATEX pwfx, WORD wChannels, DWORD dwSamplesPerSec, WORD wBitsPerSample)
  53. {
  54. pwfx->wFormatTag = WAVE_FORMAT_PCM;
  55. pwfx->nChannels = min(2, max(1, wChannels));
  56. pwfx->nSamplesPerSec = min(DSBFREQUENCY_MAX, max(DSBFREQUENCY_MIN, dwSamplesPerSec));
  57. if(wBitsPerSample < 12)
  58. {
  59. pwfx->wBitsPerSample = 8;
  60. }
  61. else
  62. {
  63. pwfx->wBitsPerSample = 16;
  64. }
  65. pwfx->nBlockAlign = pwfx->nChannels * pwfx->wBitsPerSample / 8;
  66. pwfx->nAvgBytesPerSec = pwfx->nSamplesPerSec * pwfx->nBlockAlign;
  67. pwfx->cbSize = 0;
  68. }
  69. #if 0
  70. HRESULT InternalCreateSoundBuffer(LPDSBUFFERDESC pDsbDesc, byte *pbWaveData,DWORD cbWaveData,LPDIRECTSOUND lpDirectSound, LPDIRECTSOUNDBUFFER *lplpDirectSoundBuffer)
  71. {
  72. HRESULT hr = DS_OK;
  73. HACMSTREAM has = NULL;
  74. BOOL fPrep = FALSE;
  75. ACMSTREAMHEADER ash;
  76. DWORD dwBufferBytes;
  77. LPVOID pvWrite;
  78. DWORD cbWrite;
  79. HMMIO hmm = NULL;
  80. MMRESULT mmr;
  81. MMIOINFO mmioinfo;
  82. MMCKINFO ckiRiff;
  83. MMCKINFO cki;
  84. LPWAVEFORMATEX pwfxSrcFormat = NULL;
  85. LPWAVEFORMATEX pwfxDestFormat = NULL;
  86. BOOL bNULLFORMAT = FALSE;
  87. ZeroMemory(&mmioinfo, sizeof(mmioinfo));
  88. if(SUCCEEDED(hr)){
  89. mmioinfo.fccIOProc = FOURCC_MEM;
  90. mmioinfo.pchBuffer = (HPSTR)pbWaveData;
  91. mmioinfo.cchBuffer = cbWaveData;
  92. hmm = mmioOpen(NULL, &mmioinfo, MMIO_READ);
  93. if(!hmm)
  94. {
  95. DPF1(1, "Unable to open file via MMIO. Error %lu", mmioinfo.wErrorRet);
  96. hr = E_FAIL;
  97. }
  98. }
  99. // Decend into the RIFF chunk
  100. if(SUCCEEDED(hr))
  101. {
  102. ckiRiff.ckid = FOURCC_RIFF;
  103. mmr = mmioDescend(hmm, &ckiRiff, NULL, MMIO_FINDCHUNK);
  104. if(MMSYSERR_NOERROR != mmr)
  105. {
  106. DPF1(1, "Unable to descend into RIFF chunk. Error %lu", mmr);
  107. hr = E_FAIL;
  108. }
  109. }
  110. // Verify that this is a wave file
  111. if(SUCCEEDED(hr) && FOURCC_WAVE != ckiRiff.fccType)
  112. {
  113. DPF1(1, "File is not type WAVE %d",GetLastError());
  114. hr = DSERR_BADFORMAT;
  115. }
  116. // Decend into the format chunk
  117. if(SUCCEEDED(hr))
  118. {
  119. cki.ckid = FOURCC_FORMAT;
  120. mmr = mmioDescend(hmm, &cki, &ckiRiff, MMIO_FINDCHUNK);
  121. if(MMSYSERR_NOERROR != mmr)
  122. {
  123. DPF1(1, "Unable to descend into format chunk. Error %lu", mmr);
  124. hr = E_FAIL; //MMRESULTtoHRESULT(mmr);
  125. }
  126. if(SUCCEEDED(hr))
  127. {
  128. pwfxSrcFormat = (LPWAVEFORMATEX)(pbWaveData + cki.dwDataOffset);
  129. }
  130. }
  131. // Ascend out of the format chunk
  132. if(SUCCEEDED(hr))
  133. {
  134. mmr = mmioAscend(hmm, &cki, 0);
  135. if(MMSYSERR_NOERROR != mmr)
  136. {
  137. DPF(1, "Unable to ascend out of format chunk. Error %lu", mmr);
  138. hr = E_FAIL; //MMRESULTtoHRESULT(mmr);
  139. }
  140. }
  141. // Descend into the data chunk
  142. if(SUCCEEDED(hr))
  143. {
  144. cki.ckid = FOURCC_DATA;
  145. mmr = mmioDescend(hmm, &cki, &ckiRiff, MMIO_FINDCHUNK);
  146. if(MMSYSERR_NOERROR != mmr)
  147. {
  148. RPF(DPFLVL_ERROR, "Unable to descend into data chunk. Error %lu", mmr);
  149. hr = E_FAIL; //MMRESULTtoHRESULT(mmr);
  150. }
  151. }
  152. // Prepare PCM conversion
  153. if(SUCCEEDED(hr))
  154. {
  155. if(WAVE_FORMAT_PCM == pwfxSrcFormat->wFormatTag)
  156. {
  157. // Populate the buffer description
  158. dwBufferBytes = cki.cksize;
  159. pwfxDestFormat = pwfxSrcFormat;
  160. }
  161. else
  162. {
  163. // Open an ACM conversion stream
  164. mmr = acmStreamOpen(&has, NULL, (LPWAVEFORMATEX)pwfxSrcFormat, pwfxDestFormat, NULL, 0, 0, 0);
  165. if(MMSYSERR_NOERROR != mmr)
  166. {
  167. RPF(DPFLVL_ERROR, "Unable to open an ACM stream. Error %lu", mmr);
  168. hr = E_FAIL; //MMRESULTtoHRESULT(mmr);
  169. }
  170. // Get the size of the PCM data
  171. if(SUCCEEDED(hr))
  172. {
  173. mmr = acmStreamSize(has, cki.cksize, &dwBufferBytes, ACM_STREAMSIZEF_SOURCE);
  174. if(MMSYSERR_NOERROR != mmr)
  175. {
  176. RPF(DPFLVL_ERROR, "Unable to determine converted data size. Error %lu", mmr);
  177. hr = E_FAIL; //MMRESULTtoHRESULT(mmr);
  178. }
  179. }
  180. // Create the destination format
  181. if(SUCCEEDED(hr))
  182. {
  183. pwfxDestFormat = (WAVEFORMATEX*)malloc(sizeof(WAVEFORMATEX));
  184. if (pwfxDestFormat==NULL) hr=E_OUTOFMEMORY;
  185. }
  186. if(SUCCEEDED(hr))
  187. {
  188. FillWfx(pwfxDestFormat, pwfxSrcFormat->nChannels, pwfxSrcFormat->nSamplesPerSec, pwfxSrcFormat->wBitsPerSample);
  189. }
  190. }
  191. }
  192. LPDIRECTSOUNDBUFFER lpDirectSoundBuffer=NULL;
  193. if(SUCCEEDED(hr))
  194. {
  195. //hr = InitializeEmpty(pDsbDesc->dwFlags, dwBufferBytes, pwfxDestFormat, NULL);
  196. pDsbDesc->dwBufferBytes=dwBufferBytes;
  197. if (pDsbDesc->lpwfxFormat){
  198. memcpy(pDsbDesc->lpwfxFormat,pwfxDestFormat,sizeof(WAVEFORMATEX));
  199. }
  200. else {
  201. pDsbDesc->lpwfxFormat=pwfxDestFormat;
  202. }
  203. hr=lpDirectSound->CreateSoundBuffer(pDsbDesc,lplpDirectSoundBuffer,NULL);
  204. if (*lplpDirectSoundBuffer==NULL) hr= E_FAIL;
  205. lpDirectSoundBuffer=*lplpDirectSoundBuffer;
  206. }
  207. // Lock the buffer in order to write the PCM data to it
  208. if(SUCCEEDED(hr))
  209. {
  210. hr = lpDirectSoundBuffer->Lock(0, dwBufferBytes, &pvWrite, &cbWrite, NULL, NULL,0);
  211. }
  212. // Convert to PCM
  213. if(SUCCEEDED(hr))
  214. {
  215. if(WAVE_FORMAT_PCM == pwfxSrcFormat->wFormatTag)
  216. {
  217. CopyMemory(pvWrite, pbWaveData + cki.dwDataOffset, cbWrite);
  218. }
  219. else
  220. {
  221. // Prepare the conversion header
  222. ZeroMemory(&ash, sizeof(ash));
  223. ash.cbStruct = sizeof(ash);
  224. ash.pbSrc = pbWaveData + cki.dwDataOffset;
  225. ash.cbSrcLength = cki.cksize;
  226. ash.pbDst = (LPBYTE)pvWrite;
  227. ash.cbDstLength = cbWrite;
  228. mmr = acmStreamPrepareHeader(has, &ash, 0);
  229. if(MMSYSERR_NOERROR != mmr)
  230. {
  231. RPF(DPFLVL_ERROR, "Unable to prepare ACM stream header. Error %lu", mmr);
  232. hr = E_FAIL; //MMRESULTtoHRESULT(mmr);
  233. }
  234. fPrep = SUCCEEDED(hr);
  235. // Convert the buffer
  236. if(SUCCEEDED(hr))
  237. {
  238. mmr = acmStreamConvert(has, &ash, 0);
  239. if(MMSYSERR_NOERROR != mmr)
  240. {
  241. RPF(DPFLVL_ERROR, "Unable to convert wave data. Error %lu", mmr);
  242. hr = E_FAIL; //MMRESULTtoHRESULT(mmr);
  243. }
  244. }
  245. }
  246. }
  247. // Unlock the buffer
  248. if(SUCCEEDED(hr))
  249. {
  250. hr = lpDirectSoundBuffer->Unlock(pvWrite, cbWrite, NULL, 0);
  251. }
  252. // Clean up
  253. if(fPrep)
  254. {
  255. acmStreamUnprepareHeader(has, &ash, 0);
  256. }
  257. if(has)
  258. {
  259. acmStreamClose(has, 0);
  260. }
  261. if(hmm)
  262. {
  263. mmioClose(hmm, 0);
  264. }
  265. if(pwfxDestFormat != pwfxSrcFormat)
  266. {
  267. free(pwfxDestFormat);
  268. }
  269. return hr;
  270. }
  271. #endif
  272. ///////////////////////////////////////////////////////////////////////////////////////////
  273. HRESULT InternalCreateSoundBuffer(LPDSBUFFERDESC pDsbDesc, byte *pbWaveData, DWORD cbWaveData,LPDIRECTSOUND lpDirectSound, LPDIRECTSOUNDBUFFER *lplpDirectSoundBuffer)
  274. {
  275. HRESULT hr = DS_OK;
  276. HACMSTREAM has = NULL;
  277. BOOL fPrep = FALSE;
  278. DWORD dwBufferBytes = 0;
  279. LPVOID pvWrite = NULL;
  280. DWORD cbWrite = 0;
  281. LPWAVEFORMATEX pwfxFormat = NULL;
  282. LPWAVEFORMATEX pwfxSrcFormat = NULL;
  283. LPWAVEFORMATEX pwfxDestFormat = NULL;
  284. MMRESULT mmr = 0;
  285. DWORD dwDataLength = 0;
  286. DWORD dwOffset = 0;
  287. char *pChunk = NULL;
  288. LPDIRECTSOUNDBUFFER lpDirectSoundBuffer = NULL;
  289. ACMSTREAMHEADER ash;
  290. BOOL bNULLFORMAT =FALSE;
  291. BOOL bDirty =FALSE;
  292. struct tag_FileHeader
  293. {
  294. DWORD dwRiff;
  295. DWORD dwFileSize;
  296. DWORD dwWave;
  297. DWORD dwFormat;
  298. DWORD dwFormatLength;
  299. } FileHeader;
  300. ZeroMemory(&FileHeader,sizeof(struct tag_FileHeader));
  301. // If our file is big enough to have a header copy it over
  302. // other wise error out
  303. if (cbWaveData>sizeof(struct tag_FileHeader))
  304. {
  305. memcpy(&FileHeader,pbWaveData,sizeof(struct tag_FileHeader));
  306. }
  307. else
  308. {
  309. hr= E_INVALIDARG;
  310. }
  311. // File must be a riff file ( 52 R, 49 I, 46 F, 46 F)
  312. if (FileHeader.dwRiff != 0x46464952)
  313. {
  314. DPF(1, "DXVB: not a RIFF file\n");
  315. return E_INVALIDARG;
  316. }
  317. // must be a WAVE format ( 57 W, 41 A, 56 V, 45 E )
  318. if (FileHeader.dwWave != 0x45564157)
  319. {
  320. DPF(1, "DXVB: not a WAVE file\n");
  321. return E_INVALIDARG;
  322. }
  323. // check for odd stuff
  324. // note 18bytes is a typical WAVEFORMATEX
  325. if (FileHeader.dwFormatLength <= 14) return E_INVALIDARG;
  326. if (FileHeader.dwFormatLength > 1000) return E_INVALIDARG;
  327. //allocate the waveformat
  328. pwfxFormat=(WAVEFORMATEX*)alloca(FileHeader.dwFormatLength);
  329. if (!pwfxFormat) return E_OUTOFMEMORY;
  330. //copy it to our own data structure
  331. pChunk=(char*)(pbWaveData+sizeof (struct tag_FileHeader));
  332. memcpy(pwfxFormat,pChunk,FileHeader.dwFormatLength);
  333. // Now look for the next chunk after the WaveFormat
  334. pChunk=(char*)(pChunk+FileHeader.dwFormatLength);
  335. // Look for option FACT chunk and skip it
  336. // (66 F, 61 A, 63 C, 74 T)
  337. // this chunk is required for compressed wave files
  338. // but is optional for PCM
  339. //
  340. if ( ((DWORD*)pChunk)[0]==0x74636166)
  341. {
  342. dwOffset=((DWORD*)pChunk)[1];
  343. dwBufferBytes=((DWORD*)pChunk)[2]; //number of bytes of PCM data
  344. pChunk =(char*)(pChunk+ dwOffset+8);
  345. }
  346. //Look for required data chunk
  347. // (64 D, 61 A, 74 T, 61 A)
  348. if (((DWORD*)pChunk)[0]!=0x61746164)
  349. {
  350. DPF(1, "DXVB: no DATA chunk in wave file\n");
  351. return E_INVALIDARG;
  352. }
  353. dwDataLength=((DWORD*)pChunk)[1];
  354. pChunk=(char*)(pChunk+8);
  355. //IF we assume PCM
  356. //pcm files are not required to have their fact chunk
  357. //so be ware they may missreport the data length
  358. dwBufferBytes=dwDataLength;
  359. pwfxDestFormat=pwfxSrcFormat=pwfxFormat;
  360. // if we are not PCM then we need to do some things first
  361. if (pwfxFormat->wFormatTag!=WAVE_FORMAT_PCM)
  362. {
  363. // source format is from the file
  364. pwfxSrcFormat=pwfxFormat; //from file
  365. pwfxDestFormat=pDsbDesc->lpwfxFormat ; //from user
  366. //pick the format of the file passed in
  367. FillWfx(pwfxDestFormat, pwfxSrcFormat->nChannels, pwfxSrcFormat->nSamplesPerSec, pwfxSrcFormat->wBitsPerSample);
  368. // Open an ACM conversion stream
  369. mmr = acmStreamOpen(&has, NULL, (LPWAVEFORMATEX)pwfxSrcFormat, pwfxDestFormat, NULL, 0, 0, ACM_STREAMOPENF_NONREALTIME );
  370. if(MMSYSERR_NOERROR != mmr)
  371. {
  372. DPF1(1, "Unable to open an ACM stream. Error %lu\n", mmr);
  373. return E_FAIL;
  374. }
  375. // Get the size of the PCM data
  376. mmr = acmStreamSize(has, dwDataLength, &dwBufferBytes, ACM_STREAMSIZEF_SOURCE);
  377. if(MMSYSERR_NOERROR != mmr)
  378. {
  379. DPF1(1, "Unable to determine converted data size. Error %lu\n", mmr);
  380. return E_FAIL; //MMRESULTtoHRESULT(mmr);
  381. }
  382. // Allocate a DestFormat struct
  383. //pwfxDestFormat = (WAVEFORMATEX*)alloca(sizeof(WAVEFORMATEX));
  384. //if (!pwfxDestFormat) return E_OUTOFMEMORY;
  385. // Fill the format with information from the source but
  386. // FillWfx sets the format to PCM
  387. //FillWfx(pwfxDestFormat, pwfxSrcFormat->nChannels, pwfxSrcFormat->nSamplesPerSec, pwfxSrcFormat->wBitsPerSample);
  388. }
  389. // fill the buffer desc the user passed in with the buffer bytes
  390. // this is the number of PCM bytes
  391. pDsbDesc->dwBufferBytes=dwBufferBytes;
  392. // if they provide us a pointer to a waveformatex
  393. // copy over the format to the input desc and use it
  394. // otherwise have it point to our data format temprarily
  395. if (pDsbDesc->lpwfxFormat){
  396. memcpy(pDsbDesc->lpwfxFormat,pwfxDestFormat,sizeof(WAVEFORMATEX));
  397. }
  398. else {
  399. pDsbDesc->lpwfxFormat=pwfxDestFormat;
  400. //make sure we null out the format before passing it back to the user
  401. //NOTE: consider the problems in a multithreaded enviroment
  402. //where the users data structures are being accesed by multiple
  403. //threads... on the other hand if thats going on..
  404. //then the user would need to syncronize things on his or her own
  405. //for everything else including calling into apis that fill structures..
  406. bNULLFORMAT=TRUE;
  407. }
  408. // Create the buffer
  409. hr=lpDirectSound->CreateSoundBuffer(pDsbDesc,lplpDirectSoundBuffer,NULL);
  410. if FAILED(hr) return hr;
  411. if (*lplpDirectSoundBuffer==NULL) return E_FAIL; //todo ASSERT this instead..
  412. // for more convenient referencing...
  413. lpDirectSoundBuffer=*lplpDirectSoundBuffer;
  414. // Lock the buffer in order to write the PCM data to it
  415. // cbWrite will contain the number of locked bytes
  416. hr = lpDirectSoundBuffer->Lock(0, dwBufferBytes, &pvWrite, &cbWrite, NULL, NULL,0);
  417. if FAILED(hr) return hr;
  418. // If the sorce format was pcm then copy from the file to the buffer
  419. if(WAVE_FORMAT_PCM == pwfxSrcFormat->wFormatTag)
  420. {
  421. CopyMemory(pvWrite, pChunk, cbWrite);
  422. // Unlock the buffer
  423. hr = lpDirectSoundBuffer->Unlock(pvWrite, cbWrite, NULL, 0);
  424. if (FAILED(hr))
  425. {
  426. DPF(1, "DXVB: lpDirectSoundBuffer->Unlock failed.. \n");
  427. return hr;
  428. }
  429. }
  430. // if the source format is compressed then convert first then copy
  431. else
  432. {
  433. // Prepare the conversion header
  434. ZeroMemory(&ash, sizeof(ash));
  435. ash.cbStruct = sizeof(ash);
  436. ash.pbSrc = (unsigned char*)pChunk; //start of compressed data
  437. ash.cbSrcLength = dwDataLength; //number of bytes of compressed data
  438. ash.pbDst = (LPBYTE)pvWrite; //where to put the decompressed data
  439. ash.cbDstLength = cbWrite; //how big is that buffer
  440. mmr = acmStreamPrepareHeader(has, &ash, 0);
  441. if(MMSYSERR_NOERROR != mmr)
  442. {
  443. DPF1(1, "DXVB: Unable to prepare ACM stream header. Error %lu \n", mmr);
  444. return E_FAIL;
  445. }
  446. mmr = acmStreamConvert(has, &ash, 0);
  447. if(MMSYSERR_NOERROR != mmr)
  448. {
  449. DPF1(1, "DXVB: Unable to convert wave data. Error %lu \n", mmr);
  450. return hr;
  451. }
  452. // Unlock the buffer
  453. hr = lpDirectSoundBuffer->Unlock(pvWrite, cbWrite, NULL, 0);
  454. if (FAILED(hr))
  455. {
  456. DPF(1, "DXVB: lpDirectSoundBuffer->Unlock failed.. \n");
  457. return hr;
  458. }
  459. acmStreamUnprepareHeader(has, &ash, 0);
  460. acmStreamClose(has, 0);
  461. }
  462. if (bNULLFORMAT){
  463. pDsbDesc->lpwfxFormat=NULL;
  464. }
  465. return hr;
  466. }
  467. HRESULT InternalCreateSoundBufferFromFile(LPDIRECTSOUND lpDirectSound,LPDSBUFFERDESC pDesc,WCHAR *file,LPDIRECTSOUNDBUFFER *lplpDirectSoundBuffer)
  468. {
  469. HRESULT hr=S_OK;
  470. HANDLE hFile = NULL;
  471. HANDLE hFileMapping = NULL;
  472. DWORD cbWaveData;
  473. LPBYTE pbWaveData = NULL;
  474. #pragma message("CreateFileW should be used for localization why wont it work")
  475. //hFile = CreateFileW(file, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL | FILE_FLAG_SEQUENTIAL_SCAN, NULL);
  476. USES_CONVERSION;
  477. LPSTR pStrA=W2T(file);
  478. if (!pStrA) return E_INVALIDARG;
  479. hFile = CreateFileA(pStrA, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL | FILE_FLAG_SEQUENTIAL_SCAN, NULL);
  480. if(INVALID_HANDLE_VALUE == hFile)
  481. {
  482. hFile = NULL;
  483. }
  484. if(!hFile)
  485. {
  486. RPF(DPFLVL_ERROR, "Unable to open file. Error %lu", GetLastError());
  487. hr=STG_E_FILENOTFOUND;
  488. return hr;
  489. }
  490. if(hFile)
  491. {
  492. cbWaveData = GetFileSize(hFile, NULL);
  493. if(-1 == cbWaveData)
  494. {
  495. RPF(DPFLVL_ERROR, "Unable to get file size. Error %lu", GetLastError());
  496. hr = E_FAIL; //DSERR_FILEREADFAULT;
  497. }
  498. }
  499. if(SUCCEEDED(hr))
  500. {
  501. hFileMapping = CreateFileMapping(hFile, NULL, PAGE_READONLY, 0, cbWaveData, NULL);
  502. if(INVALID_HANDLE_VALUE == hFileMapping)
  503. {
  504. hFileMapping = NULL;
  505. }
  506. if(!hFileMapping)
  507. {
  508. RPF(DPFLVL_ERROR, "Unable to create file mapping. Error %lu", GetLastError());
  509. hr = E_FAIL; //DSERR_FILEREADFAULT;
  510. }
  511. }
  512. if(SUCCEEDED(hr))
  513. {
  514. pbWaveData = (LPBYTE)MapViewOfFile(hFileMapping, FILE_MAP_READ, 0, 0, cbWaveData);
  515. if(!pbWaveData)
  516. {
  517. RPF(DPFLVL_ERROR, "Unable to map view of file. Error %lu", GetLastError());
  518. hr = E_FAIL; //DSERR_FILEREADFAULT;
  519. }
  520. }
  521. if(SUCCEEDED(hr)) {
  522. hr=InternalCreateSoundBuffer(pDesc, pbWaveData, cbWaveData,lpDirectSound, lplpDirectSoundBuffer);
  523. }
  524. if(pbWaveData)
  525. {
  526. UnmapViewOfFile(pbWaveData);
  527. }
  528. if(hFileMapping)
  529. {
  530. CloseHandle(hFileMapping);
  531. }
  532. if(hFile)
  533. {
  534. CloseHandle(hFile);
  535. }
  536. return hr;
  537. }
  538. HRESULT InternalCreateSoundBufferFromResource(LPDIRECTSOUND lpDirectSound,LPDSBUFFERDESC pDesc,HANDLE resHandle,WCHAR *resName,LPDIRECTSOUNDBUFFER *lplpDirectSoundBuffer)
  539. {
  540. const LPCSTR apszResourceTypeA[] = { "WAVE", "WAV" };
  541. const LPCWSTR apszResourceTypeW[] = { L"WAVE", L"WAV" };
  542. UINT cResourceType = 2;
  543. HRSRC hRsrc = NULL;
  544. DWORD cbWaveData;
  545. LPBYTE pbWaveData = NULL;
  546. HRESULT hr=S_OK;
  547. LPCDSBUFFERDESC pDsbDesc=pDesc;
  548. while(!hRsrc && cResourceType--)
  549. {
  550. hRsrc = FindResourceW((HINSTANCE)resHandle, resName, apszResourceTypeW[cResourceType]);
  551. }
  552. if(!hRsrc)
  553. {
  554. RPF(DPFLVL_ERROR,"Unable to find resource. Error %lu", GetLastError());
  555. hr = STG_E_FILENOTFOUND;
  556. }
  557. if(SUCCEEDED(hr))
  558. {
  559. cbWaveData = SizeofResource((HINSTANCE)resHandle, hRsrc);
  560. if(!cbWaveData)
  561. {
  562. RPF(DPFLVL_ERROR, "Unable to get resource size. Error %lu", GetLastError());
  563. hr = E_FAIL;
  564. }
  565. }
  566. if(SUCCEEDED(hr))
  567. {
  568. pbWaveData = (LPBYTE)LoadResource((HINSTANCE)resHandle, hRsrc);
  569. if(!pbWaveData)
  570. {
  571. RPF(DPFLVL_ERROR, "Unable to load resource. Error %lu", GetLastError());
  572. hr = E_FAIL;
  573. }
  574. }
  575. if(SUCCEEDED(hr)) {
  576. hr=InternalCreateSoundBuffer(pDesc, pbWaveData, cbWaveData,lpDirectSound, lplpDirectSoundBuffer);
  577. }
  578. //loadResource
  579. return hr;
  580. }
  581. HRESULT InternalSaveToFile(IDirectSoundBuffer *pBuff,BSTR file)
  582. {
  583. WAVEFORMATEX waveFormat;
  584. DWORD dwWritten=0;
  585. DWORD dwBytes=0;
  586. LPBYTE lpByte=NULL;
  587. HRESULT hr;
  588. HANDLE hFile=NULL;
  589. if (!pBuff) return E_FAIL;
  590. if (!file) return E_INVALIDARG;
  591. pBuff->GetFormat(&waveFormat,sizeof(WAVEFORMATEX),NULL);
  592. USES_CONVERSION;
  593. LPSTR pStrA=W2T(file);
  594. hFile = CreateFile
  595. (
  596. pStrA,
  597. GENERIC_WRITE,
  598. 0,
  599. NULL,
  600. CREATE_ALWAYS,
  601. FILE_ATTRIBUTE_NORMAL,
  602. NULL
  603. );
  604. if (INVALID_HANDLE_VALUE != hFile)
  605. {
  606. struct tag_FileHeader
  607. {
  608. DWORD dwRiff;
  609. DWORD dwFileSize;
  610. DWORD dwWave;
  611. DWORD dwFormat;
  612. DWORD dwFormatLength;
  613. WORD wFormatTag;
  614. WORD nChannels;
  615. DWORD nSamplesPerSec;
  616. DWORD nAvgBytesPerSec;
  617. WORD nBlockAlign;
  618. WORD wBitsPerSample;
  619. DWORD dwData;
  620. DWORD dwDataLength;
  621. } FileHeader;
  622. hr=pBuff->Lock(0,0,(void**)&lpByte,&dwBytes,NULL,NULL,DSBLOCK_ENTIREBUFFER);
  623. if FAILED(hr) {
  624. CloseHandle(hFile);
  625. return hr;
  626. }
  627. FileHeader.dwRiff = 0x46464952; // RIFF
  628. FileHeader.dwWave = 0x45564157; // WAVE
  629. FileHeader.dwFormat = 0x20746D66; // fmt_chnk
  630. FileHeader.dwFormatLength = 16;
  631. FileHeader.wFormatTag = WAVE_FORMAT_PCM;
  632. FileHeader.nChannels = waveFormat.nChannels ;
  633. FileHeader.nSamplesPerSec = waveFormat.nSamplesPerSec ;
  634. FileHeader.wBitsPerSample = waveFormat.wBitsPerSample ;
  635. FileHeader.nBlockAlign = FileHeader.wBitsPerSample / 8 * FileHeader.nChannels;
  636. FileHeader.nAvgBytesPerSec = FileHeader.nSamplesPerSec * FileHeader.nBlockAlign;
  637. FileHeader.dwData = 0x61746164; // data_chnk
  638. FileHeader.dwDataLength = dwBytes;
  639. FileHeader.dwFileSize = dwBytes + sizeof(FileHeader);
  640. WriteFile(hFile, &FileHeader, sizeof(FileHeader), &dwWritten, NULL);
  641. WriteFile(hFile, lpByte, dwBytes, &dwWritten, NULL);
  642. hr=pBuff->Unlock(lpByte,0,NULL,0);
  643. CloseHandle(hFile);
  644. }
  645. else{
  646. return E_FAIL;
  647. }
  648. return S_OK;
  649. }