w64.c
上传用户:shw771010
上传日期:2022-01-05
资源大小:991k
文件大小:20k
源码类别:

Audio

开发平台:

Unix_Linux

  1. /*
  2. ** Copyright (C) 1999-2009 Erik de Castro Lopo <erikd@mega-nerd.com>
  3. **
  4. ** This program is free software; you can redistribute it and/or modify
  5. ** it under the terms of the GNU Lesser General Public License as published by
  6. ** the Free Software Foundation; either version 2.1 of the License, or
  7. ** (at your option) any later version.
  8. **
  9. ** This program is distributed in the hope that it will be useful,
  10. ** but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. ** MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  12. ** GNU Lesser General Public License for more details.
  13. **
  14. ** You should have received a copy of the GNU Lesser General Public License
  15. ** along with this program; if not, write to the Free Software
  16. ** Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
  17. */
  18. #include "sfconfig.h"
  19. #include <stdio.h>
  20. #include <string.h>
  21. #include <ctype.h>
  22. #include <time.h>
  23. #include "sndfile.h"
  24. #include "sfendian.h"
  25. #include "common.h"
  26. #include "wav_w64.h"
  27. /*------------------------------------------------------------------------------
  28. ** W64 files use 16 byte markers as opposed to the four byte marker of
  29. ** WAV files.
  30. ** For comparison purposes, an integer is required, so make an integer
  31. ** hash for the 16 bytes using MAKE_HASH16 macro, but also create a 16
  32. ** byte array containing the complete 16 bytes required when writing the
  33. ** header.
  34. */
  35. #define MAKE_HASH16(x0,x1,x2,x3,x4,x5,x6,x7,x8,x9,xa,xb,xc,xd,xe,xf)
  36. ( (x0) ^ ((x1) << 1) ^ ((x2) << 2) ^ ((x3) << 3) ^
  37. ((x4) << 4)  ^ ((x5) << 5) ^ ((x6) << 6) ^ ((x7) << 7) ^
  38. ((x8) << 8)  ^ ((x9) << 9) ^ ((xa) << 10) ^ ((xb) << 11) ^ 
  39. ((xc) << 12)  ^ ((xd) << 13) ^ ((xe) << 14) ^ ((xf) << 15) )
  40. #define MAKE_MARKER16(name,x0,x1,x2,x3,x4,x5,x6,x7,x8,x9,xa,xb,xc,xd,xe,xf)
  41. static unsigned char name [16] = { (x0), (x1), (x2), (x3), (x4), (x5), 
  42. (x6), (x7), (x8), (x9), (xa), (xb), (xc), (xd), (xe), (xf) }
  43. #define riff_HASH16 MAKE_HASH16 ('r', 'i', 'f', 'f', 0x2E, 0x91, 0xCF, 0x11, 
  44. 0xA5, 0xD6, 0x28, 0xDB, 0x04, 0xC1, 0x00, 0x00)
  45. #define wave_HASH16  MAKE_HASH16 ('w', 'a', 'v', 'e', 0xF3, 0xAC, 0xD3, 0x11, 
  46. 0x8C, 0xD1, 0x00, 0xC0, 0x4F, 0x8E, 0xDB, 0x8A)
  47. #define fmt_HASH16  MAKE_HASH16 ('f', 'm', 't', ' ', 0xF3, 0xAC, 0xD3, 0x11, 
  48. 0x8C, 0xD1, 0x00, 0xC0, 0x4F, 0x8E, 0xDB, 0x8A)
  49. #define fact_HASH16  MAKE_HASH16 ('f', 'a', 'c', 't', 0xF3, 0xAC, 0xD3, 0x11, 
  50. 0x8C, 0xD1, 0x00, 0xC0, 0x4F, 0x8E, 0xDB, 0x8A)
  51. #define data_HASH16  MAKE_HASH16 ('d', 'a', 't', 'a', 0xF3, 0xAC, 0xD3, 0x11, 
  52. 0x8C, 0xD1, 0x00, 0xC0, 0x4F, 0x8E, 0xDB, 0x8A)
  53. #define ACID_HASH16  MAKE_HASH16 (0x6D, 0x07, 0x1C, 0xEA, 0xA3, 0xEF, 0x78, 0x4C, 
  54. 0x90, 0x57, 0x7F, 0x79, 0xEE, 0x25, 0x2A, 0xAE)
  55. #define levl_HASH16 MAKE_HASH16 (0x6c, 0x65, 0x76, 0x6c, 0xf3, 0xac, 0xd3, 0x11, 
  56. 0xd1, 0x8c, 0x00, 0xC0, 0x4F, 0x8E, 0xDB, 0x8A)
  57. #define list_HASH16 MAKE_HASH16 (0x6C, 0x69, 0x73, 0x74, 0x2F, 0x91, 0xCF, 0x11, 
  58. 0xA5, 0xD6, 0x28, 0xDB, 0x04, 0xC1, 0x00, 0x00)
  59. #define junk_HASH16 MAKE_HASH16 (0x6A, 0x75, 0x6E, 0x6b, 0xF3, 0xAC, 0xD3, 0x11, 
  60. 0x8C, 0xD1, 0x00, 0xC0, 0x4f, 0x8E, 0xDB, 0x8A)
  61. #define bext_MARKER MAKE_HASH16 (0x62, 0x65, 0x78, 0x74, 0xf3, 0xac, 0xd3, 0xaa, 
  62. 0xd1, 0x8c, 0x00, 0xC0, 0x4F, 0x8E, 0xDB, 0x8A)
  63. #define MARKER_HASH16 MAKE_HASH16 (0x56, 0x62, 0xf7, 0xab, 0x2d, 0x39, 0xd2, 0x11, 
  64. 0x86, 0xc7, 0x00, 0xc0, 0x4f, 0x8e, 0xdb, 0x8a)
  65. #define SUMLIST_HASH16 MAKE_HASH16 (0xBC, 0x94, 0x5F, 0x92, 0x5A, 0x52, 0xD2, 0x11, 
  66. 0x86, 0xDC, 0x00, 0xC0, 0x4F, 0x8E, 0xDB, 0x8A)
  67. MAKE_MARKER16 (riff_MARKER16, 'r', 'i', 'f', 'f', 0x2E, 0x91, 0xCF, 0x11,
  68. 0xA5, 0xD6, 0x28, 0xDB, 0x04, 0xC1, 0x00, 0x00) ;
  69. MAKE_MARKER16 (wave_MARKER16, 'w', 'a', 'v', 'e', 0xF3, 0xAC, 0xD3, 0x11,
  70. 0x8C, 0xD1, 0x00, 0xC0, 0x4F, 0x8E, 0xDB, 0x8A) ;
  71. MAKE_MARKER16 (fmt_MARKER16, 'f', 'm', 't', ' ', 0xF3, 0xAC, 0xD3, 0x11,
  72. 0x8C, 0xD1, 0x00, 0xC0, 0x4F, 0x8E, 0xDB, 0x8A) ;
  73. MAKE_MARKER16 (fact_MARKER16, 'f', 'a', 'c', 't', 0xF3, 0xAC, 0xD3, 0x11,
  74. 0x8C, 0xD1, 0x00, 0xC0, 0x4F, 0x8E, 0xDB, 0x8A) ;
  75. MAKE_MARKER16 (data_MARKER16, 'd', 'a', 't', 'a', 0xF3, 0xAC, 0xD3, 0x11,
  76. 0x8C, 0xD1, 0x00, 0xC0, 0x4F, 0x8E, 0xDB, 0x8A) ;
  77. enum
  78. { HAVE_riff = 0x01,
  79. HAVE_wave = 0x02,
  80. HAVE_fmt = 0x04,
  81. HAVE_fact = 0x08,
  82. HAVE_data = 0x20
  83. } ;
  84. /*------------------------------------------------------------------------------
  85.  * Private static functions.
  86.  */
  87. static int w64_read_header (SF_PRIVATE *psf, int *blockalign, int *framesperblock) ;
  88. static int w64_write_header (SF_PRIVATE *psf, int calc_length) ;
  89. static int w64_close (SF_PRIVATE *psf) ;
  90. /*------------------------------------------------------------------------------
  91. ** Public function.
  92. */
  93. int
  94. w64_open (SF_PRIVATE *psf)
  95. { WAV_PRIVATE * wpriv ;
  96. int subformat, error, blockalign = 0, framesperblock = 0 ;
  97. if ((wpriv = calloc (1, sizeof (WAV_PRIVATE))) == NULL)
  98. return SFE_MALLOC_FAILED ;
  99. psf->container_data = wpriv ;
  100. if (psf->file.mode == SFM_READ || (psf->file.mode == SFM_RDWR &&psf->filelength > 0))
  101. { if ((error = w64_read_header (psf, &blockalign, &framesperblock)))
  102. return error ;
  103. } ;
  104. if ((SF_CONTAINER (psf->sf.format)) != SF_FORMAT_W64)
  105. return SFE_BAD_OPEN_FORMAT ;
  106. subformat = SF_CODEC (psf->sf.format) ;
  107. if (psf->file.mode == SFM_WRITE || psf->file.mode == SFM_RDWR)
  108. { if (psf->is_pipe)
  109. return SFE_NO_PIPE_WRITE ;
  110. psf->endian = SF_ENDIAN_LITTLE ; /* All W64 files are little endian. */
  111. psf->blockwidth = psf->bytewidth * psf->sf.channels ;
  112. if (subformat == SF_FORMAT_IMA_ADPCM || subformat == SF_FORMAT_MS_ADPCM)
  113. { blockalign = wav_w64_srate2blocksize (psf->sf.samplerate * psf->sf.channels) ;
  114. framesperblock = -1 ;
  115. /* FIXME : This block must go */
  116. psf->filelength = SF_COUNT_MAX ;
  117. psf->datalength = psf->filelength ;
  118. if (psf->sf.frames <= 0)
  119. psf->sf.frames = (psf->blockwidth) ? psf->filelength / psf->blockwidth : psf->filelength ;
  120. /* EMXIF : This block must go */
  121. } ;
  122. if ((error = w64_write_header (psf, SF_FALSE)))
  123. return error ;
  124. psf->write_header = w64_write_header ;
  125. } ;
  126. psf->container_close = w64_close ;
  127. switch (subformat)
  128. { case SF_FORMAT_PCM_U8 :
  129. error = pcm_init (psf) ;
  130. break ;
  131. case SF_FORMAT_PCM_16 :
  132. case SF_FORMAT_PCM_24 :
  133. case SF_FORMAT_PCM_32 :
  134. error = pcm_init (psf) ;
  135. break ;
  136. case SF_FORMAT_ULAW :
  137. error = ulaw_init (psf) ;
  138. break ;
  139. case SF_FORMAT_ALAW :
  140. error = alaw_init (psf) ;
  141. break ;
  142. /* Lite remove start */
  143. case SF_FORMAT_FLOAT :
  144. error = float32_init (psf) ;
  145. break ;
  146. case SF_FORMAT_DOUBLE :
  147. error = double64_init (psf) ;
  148. break ;
  149. case SF_FORMAT_IMA_ADPCM :
  150. error = wav_w64_ima_init (psf, blockalign, framesperblock) ;
  151. break ;
  152. case SF_FORMAT_MS_ADPCM :
  153. error = wav_w64_msadpcm_init (psf, blockalign, framesperblock) ;
  154. break ;
  155. /* Lite remove end */
  156. case SF_FORMAT_GSM610 :
  157. error = gsm610_init (psf) ;
  158. break ;
  159. default :  return SFE_UNIMPLEMENTED ;
  160. } ;
  161. return error ;
  162. } /* w64_open */
  163. /*=========================================================================
  164. ** Private functions.
  165. */
  166. static int
  167. w64_read_header (SF_PRIVATE *psf, int *blockalign, int *framesperblock)
  168. { WAV_PRIVATE *wpriv ;
  169. WAV_FMT  *wav_fmt ;
  170. int dword = 0, marker, format = 0 ;
  171. sf_count_t chunk_size, bytesread = 0 ;
  172. int parsestage = 0, error, done = 0 ;
  173. if ((wpriv = psf->container_data) == NULL)
  174. return SFE_INTERNAL ;
  175. wav_fmt = &wpriv->wav_fmt ;
  176. /* Set position to start of file to begin reading header. */
  177. psf_binheader_readf (psf, "p", 0) ;
  178. while (! done)
  179. { /* Each new chunk must start on an 8 byte boundary, so jump if needed. */
  180. if (psf->headindex & 0x7)
  181. psf_binheader_readf (psf, "j", 8 - (psf->headindex & 0x7)) ;
  182. /* Generate hash of 16 byte marker. */
  183. bytesread += psf_binheader_readf (psf, "h", &marker) ;
  184. chunk_size = 0 ;
  185. switch (marker)
  186. { case riff_HASH16 :
  187. if (parsestage)
  188. return SFE_W64_NO_RIFF ;
  189. bytesread += psf_binheader_readf (psf, "e8", &chunk_size) ;
  190. if (psf->filelength != chunk_size)
  191. psf_log_printf (psf, "riff : %D (should be %D)n", chunk_size, psf->filelength) ;
  192. else
  193. psf_log_printf (psf, "riff : %Dn", chunk_size) ;
  194. parsestage |= HAVE_riff ;
  195. break ;
  196. case ACID_HASH16:
  197. psf_log_printf (psf, "Looks like an ACID file. Exiting.n") ;
  198. return SFE_UNIMPLEMENTED ;
  199. case wave_HASH16 :
  200. if ((parsestage & HAVE_riff) != HAVE_riff)
  201. return SFE_W64_NO_WAVE ;
  202. psf_log_printf (psf, "waven") ;
  203. parsestage |= HAVE_wave ;
  204. break ;
  205. case fmt_HASH16 :
  206. if ((parsestage & (HAVE_riff | HAVE_wave)) != (HAVE_riff | HAVE_wave))
  207. return SFE_WAV_NO_FMT ;
  208. bytesread += psf_binheader_readf (psf, "e8", &chunk_size) ;
  209. psf_log_printf (psf, " fmt : %Dn", chunk_size) ;
  210. /* size of 16 byte marker and 8 byte chunk_size value. */
  211. chunk_size -= 24 ;
  212. if ((error = wav_w64_read_fmt_chunk (psf, (int) chunk_size)))
  213. return error ;
  214. if (chunk_size % 8)
  215. psf_binheader_readf (psf, "j", 8 - (chunk_size % 8)) ;
  216. format = wav_fmt->format ;
  217. parsestage |= HAVE_fmt ;
  218. break ;
  219. case fact_HASH16:
  220. { sf_count_t frames ;
  221. psf_binheader_readf (psf, "e88", &chunk_size, &frames) ;
  222. psf_log_printf (psf, "   fact : %Dn     frames : %Dn",
  223. chunk_size, frames) ;
  224. } ;
  225. break ;
  226. case data_HASH16 :
  227. if ((parsestage & (HAVE_riff | HAVE_wave | HAVE_fmt)) != (HAVE_riff | HAVE_wave | HAVE_fmt))
  228. return SFE_W64_NO_DATA ;
  229. psf_binheader_readf (psf, "e8", &chunk_size) ;
  230. psf->dataoffset = psf_ftell (psf) ;
  231. psf->datalength = chunk_size - 24 ;
  232. if (chunk_size % 8)
  233. chunk_size += 8 - (chunk_size % 8) ;
  234. psf_log_printf (psf, "data : %Dn", chunk_size) ;
  235. parsestage |= HAVE_data ;
  236. if (! psf->sf.seekable)
  237. break ;
  238. /* Seek past data and continue reading header. */
  239. psf_fseek (psf, chunk_size, SEEK_CUR) ;
  240. break ;
  241. case levl_HASH16 :
  242. psf_binheader_readf (psf, "e8", &chunk_size) ;
  243. psf_log_printf (psf, "levl : %Dn", chunk_size) ;
  244. dword = chunk_size ;
  245. psf_binheader_readf (psf, "j", dword - 24) ;
  246. break ;
  247. case list_HASH16 :
  248. psf_binheader_readf (psf, "e8", &chunk_size) ;
  249. psf_log_printf (psf, "list : %Dn", chunk_size) ;
  250. dword = chunk_size ;
  251. psf_binheader_readf (psf, "j", dword - 24) ;
  252. break ;
  253. case junk_HASH16 :
  254. psf_binheader_readf (psf, "e8", &chunk_size) ;
  255. psf_log_printf (psf, "junk : %Dn", chunk_size) ;
  256. dword = chunk_size ;
  257. psf_binheader_readf (psf, "j", dword - 24) ;
  258. break ;
  259. case bext_MARKER :
  260. psf_binheader_readf (psf, "e8", &chunk_size) ;
  261. psf_log_printf (psf, "bext : %Dn", chunk_size) ;
  262. dword = chunk_size ;
  263. psf_binheader_readf (psf, "j", dword - 24) ;
  264. break ;
  265. case MARKER_HASH16 :
  266. psf_binheader_readf (psf, "e8", &chunk_size) ;
  267. psf_log_printf (psf, "marker : %Dn", chunk_size) ;
  268. dword = chunk_size ;
  269. psf_binheader_readf (psf, "j", dword - 24) ;
  270. break ;
  271. case SUMLIST_HASH16 :
  272. psf_binheader_readf (psf, "e8", &chunk_size) ;
  273. psf_log_printf (psf, "summary list : %Dn", chunk_size) ;
  274. dword = chunk_size ;
  275. psf_binheader_readf (psf, "j", dword - 24) ;
  276. break ;
  277. default :
  278. psf_log_printf (psf, "*** Unknown chunk marker : %X. Exiting parser.n", marker) ;
  279. done = SF_TRUE ;
  280. break ;
  281. } ; /* switch (dword) */
  282. if (psf->sf.seekable == 0 && (parsestage & HAVE_data))
  283. break ;
  284. if (psf_ftell (psf) >= (psf->filelength - (2 * SIGNED_SIZEOF (dword))))
  285. break ;
  286. } ; /* while (1) */
  287. if (psf->dataoffset <= 0)
  288. return SFE_W64_NO_DATA ;
  289. psf->endian = SF_ENDIAN_LITTLE ; /* All W64 files are little endian. */
  290. if (psf_ftell (psf) != psf->dataoffset)
  291. psf_fseek (psf, psf->dataoffset, SEEK_SET) ;
  292. if (psf->blockwidth)
  293. { if (psf->filelength - psf->dataoffset < psf->datalength)
  294. psf->sf.frames = (psf->filelength - psf->dataoffset) / psf->blockwidth ;
  295. else
  296. psf->sf.frames = psf->datalength / psf->blockwidth ;
  297. } ;
  298. switch (format)
  299. { case WAVE_FORMAT_PCM :
  300. case WAVE_FORMAT_EXTENSIBLE :
  301. /* extensible might be FLOAT, MULAW, etc as well! */
  302. psf->sf.format = SF_FORMAT_W64 | u_bitwidth_to_subformat (psf->bytewidth * 8) ;
  303. break ;
  304. case WAVE_FORMAT_MULAW :
  305. psf->sf.format = (SF_FORMAT_W64 | SF_FORMAT_ULAW) ;
  306. break ;
  307. case WAVE_FORMAT_ALAW :
  308. psf->sf.format = (SF_FORMAT_W64 | SF_FORMAT_ALAW) ;
  309. break ;
  310. case WAVE_FORMAT_MS_ADPCM :
  311. psf->sf.format = (SF_FORMAT_W64 | SF_FORMAT_MS_ADPCM) ;
  312. *blockalign = wav_fmt->msadpcm.blockalign ;
  313. *framesperblock = wav_fmt->msadpcm.samplesperblock ;
  314. break ;
  315. case WAVE_FORMAT_IMA_ADPCM :
  316. psf->sf.format = (SF_FORMAT_W64 | SF_FORMAT_IMA_ADPCM) ;
  317. *blockalign = wav_fmt->ima.blockalign ;
  318. *framesperblock = wav_fmt->ima.samplesperblock ;
  319. break ;
  320. case WAVE_FORMAT_GSM610 :
  321. psf->sf.format = (SF_FORMAT_W64 | SF_FORMAT_GSM610) ;
  322. break ;
  323. case WAVE_FORMAT_IEEE_FLOAT :
  324. psf->sf.format = SF_FORMAT_W64 ;
  325. psf->sf.format |= (psf->bytewidth == 8) ? SF_FORMAT_DOUBLE : SF_FORMAT_FLOAT ;
  326. break ;
  327. default : return SFE_UNIMPLEMENTED ;
  328. } ;
  329. return 0 ;
  330. } /* w64_read_header */
  331. static int
  332. w64_write_header (SF_PRIVATE *psf, int calc_length)
  333. { sf_count_t  fmt_size, current ;
  334. size_t fmt_pad = 0 ;
  335. int  subformat, add_fact_chunk = SF_FALSE ;
  336. current = psf_ftell (psf) ;
  337. if (calc_length)
  338. { psf->filelength = psf_get_filelen (psf) ;
  339. psf->datalength = psf->filelength - psf->dataoffset ;
  340. if (psf->dataend)
  341. psf->datalength -= psf->filelength - psf->dataend ;
  342. if (psf->bytewidth)
  343. psf->sf.frames = psf->datalength / (psf->bytewidth * psf->sf.channels) ;
  344. } ;
  345. /* Reset the current header length to zero. */
  346. psf->header [0] = 0 ;
  347. psf->headindex = 0 ;
  348. psf_fseek (psf, 0, SEEK_SET) ;
  349. /* riff marker, length, wave and 'fmt ' markers. */
  350. psf_binheader_writef (psf, "eh8hh", riff_MARKER16, psf->filelength, wave_MARKER16, fmt_MARKER16) ;
  351. subformat = SF_CODEC (psf->sf.format) ;
  352. switch (subformat)
  353. { case SF_FORMAT_PCM_U8 :
  354. case SF_FORMAT_PCM_16 :
  355. case SF_FORMAT_PCM_24 :
  356. case SF_FORMAT_PCM_32 :
  357. fmt_size = 24 + 2 + 2 + 4 + 4 + 2 + 2 ;
  358. fmt_pad = (size_t) (8 - (fmt_size & 0x7)) ;
  359. fmt_size += fmt_pad ;
  360. /* fmt : format, channels, samplerate */
  361. psf_binheader_writef (psf, "e8224", fmt_size, WAVE_FORMAT_PCM, psf->sf.channels, psf->sf.samplerate) ;
  362. /*  fmt : bytespersec */
  363. psf_binheader_writef (psf, "e4", psf->sf.samplerate * psf->bytewidth * psf->sf.channels) ;
  364. /*  fmt : blockalign, bitwidth */
  365. psf_binheader_writef (psf, "e22", psf->bytewidth * psf->sf.channels, psf->bytewidth * 8) ;
  366. break ;
  367. case SF_FORMAT_FLOAT :
  368. case SF_FORMAT_DOUBLE :
  369. fmt_size = 24 + 2 + 2 + 4 + 4 + 2 + 2 ;
  370. fmt_pad = (size_t) (8 - (fmt_size & 0x7)) ;
  371. fmt_size += fmt_pad ;
  372. /* fmt : format, channels, samplerate */
  373. psf_binheader_writef (psf, "e8224", fmt_size, WAVE_FORMAT_IEEE_FLOAT, psf->sf.channels, psf->sf.samplerate) ;
  374. /*  fmt : bytespersec */
  375. psf_binheader_writef (psf, "e4", psf->sf.samplerate * psf->bytewidth * psf->sf.channels) ;
  376. /*  fmt : blockalign, bitwidth */
  377. psf_binheader_writef (psf, "e22", psf->bytewidth * psf->sf.channels, psf->bytewidth * 8) ;
  378. add_fact_chunk = SF_TRUE ;
  379. break ;
  380. case SF_FORMAT_ULAW :
  381. fmt_size = 24 + 2 + 2 + 4 + 4 + 2 + 2 ;
  382. fmt_pad = (size_t) (8 - (fmt_size & 0x7)) ;
  383. fmt_size += fmt_pad ;
  384. /* fmt : format, channels, samplerate */
  385. psf_binheader_writef (psf, "e8224", fmt_size, WAVE_FORMAT_MULAW, psf->sf.channels, psf->sf.samplerate) ;
  386. /*  fmt : bytespersec */
  387. psf_binheader_writef (psf, "e4", psf->sf.samplerate * psf->bytewidth * psf->sf.channels) ;
  388. /*  fmt : blockalign, bitwidth */
  389. psf_binheader_writef (psf, "e22", psf->bytewidth * psf->sf.channels, 8) ;
  390. add_fact_chunk = SF_TRUE ;
  391. break ;
  392. case SF_FORMAT_ALAW :
  393. fmt_size = 24 + 2 + 2 + 4 + 4 + 2 + 2 ;
  394. fmt_pad = (size_t) (8 - (fmt_size & 0x7)) ;
  395. fmt_size += fmt_pad ;
  396. /* fmt : format, channels, samplerate */
  397. psf_binheader_writef (psf, "e8224", fmt_size, WAVE_FORMAT_ALAW, psf->sf.channels, psf->sf.samplerate) ;
  398. /*  fmt : bytespersec */
  399. psf_binheader_writef (psf, "e4", psf->sf.samplerate * psf->bytewidth * psf->sf.channels) ;
  400. /*  fmt : blockalign, bitwidth */
  401. psf_binheader_writef (psf, "e22", psf->bytewidth * psf->sf.channels, 8) ;
  402. add_fact_chunk = SF_TRUE ;
  403. break ;
  404. /* Lite remove start */
  405. case SF_FORMAT_IMA_ADPCM :
  406. { int blockalign, framesperblock, bytespersec ;
  407. blockalign = wav_w64_srate2blocksize (psf->sf.samplerate * psf->sf.channels) ;
  408. framesperblock = 2 * (blockalign - 4 * psf->sf.channels) / psf->sf.channels + 1 ;
  409. bytespersec = (psf->sf.samplerate * blockalign) / framesperblock ;
  410. /* fmt chunk. */
  411. fmt_size = 24 + 2 + 2 + 4 + 4 + 2 + 2 + 2 + 2 ;
  412. fmt_pad = (size_t) (8 - (fmt_size & 0x7)) ;
  413. fmt_size += fmt_pad ;
  414. /* fmt : size, WAV format type, channels. */
  415. psf_binheader_writef (psf, "e822", fmt_size, WAVE_FORMAT_IMA_ADPCM, psf->sf.channels) ;
  416. /* fmt : samplerate, bytespersec. */
  417. psf_binheader_writef (psf, "e44", psf->sf.samplerate, bytespersec) ;
  418. /* fmt : blockalign, bitwidth, extrabytes, framesperblock. */
  419. psf_binheader_writef (psf, "e2222", blockalign, 4, 2, framesperblock) ;
  420. } ;
  421. add_fact_chunk = SF_TRUE ;
  422. break ;
  423. case SF_FORMAT_MS_ADPCM :
  424. { int blockalign, framesperblock, bytespersec, extrabytes ;
  425. blockalign = wav_w64_srate2blocksize (psf->sf.samplerate * psf->sf.channels) ;
  426. framesperblock = 2 + 2 * (blockalign - 7 * psf->sf.channels) / psf->sf.channels ;
  427. bytespersec = (psf->sf.samplerate * blockalign) / framesperblock ;
  428. /* fmt chunk. */
  429. extrabytes = 2 + 2 + MSADPCM_ADAPT_COEFF_COUNT * (2 + 2) ;
  430. fmt_size = 24 + 2 + 2 + 4 + 4 + 2 + 2 + 2 + extrabytes ;
  431. fmt_pad = (size_t) (8 - (fmt_size & 0x7)) ;
  432. fmt_size += fmt_pad ;
  433. /* fmt : size, W64 format type, channels. */
  434. psf_binheader_writef (psf, "e822", fmt_size, WAVE_FORMAT_MS_ADPCM, psf->sf.channels) ;
  435. /* fmt : samplerate, bytespersec. */
  436. psf_binheader_writef (psf, "e44", psf->sf.samplerate, bytespersec) ;
  437. /* fmt : blockalign, bitwidth, extrabytes, framesperblock. */
  438. psf_binheader_writef (psf, "e22222", blockalign, 4, extrabytes, framesperblock, 7) ;
  439. msadpcm_write_adapt_coeffs (psf) ;
  440. } ;
  441. add_fact_chunk = SF_TRUE ;
  442. break ;
  443. /* Lite remove end */
  444. case SF_FORMAT_GSM610 :
  445. { int bytespersec ;
  446. bytespersec = (psf->sf.samplerate * WAV_W64_GSM610_BLOCKSIZE) / WAV_W64_GSM610_SAMPLES ;
  447. /* fmt chunk. */
  448. fmt_size = 24 + 2 + 2 + 4 + 4 + 2 + 2 + 2 + 2 ;
  449. fmt_pad = (size_t) (8 - (fmt_size & 0x7)) ;
  450. fmt_size += fmt_pad ;
  451. /* fmt : size, WAV format type, channels. */
  452. psf_binheader_writef (psf, "e822", fmt_size, WAVE_FORMAT_GSM610, psf->sf.channels) ;
  453. /* fmt : samplerate, bytespersec. */
  454. psf_binheader_writef (psf, "e44", psf->sf.samplerate, bytespersec) ;
  455. /* fmt : blockalign, bitwidth, extrabytes, framesperblock. */
  456. psf_binheader_writef (psf, "e2222", WAV_W64_GSM610_BLOCKSIZE, 0, 2, WAV_W64_GSM610_SAMPLES) ;
  457. } ;
  458. add_fact_chunk = SF_TRUE ;
  459. break ;
  460. default :  return SFE_UNIMPLEMENTED ;
  461. } ;
  462. /* Pad to 8 bytes with zeros. */
  463. if (fmt_pad > 0)
  464. psf_binheader_writef (psf, "z", fmt_pad) ;
  465. if (add_fact_chunk)
  466. psf_binheader_writef (psf, "eh88", fact_MARKER16, (sf_count_t) (16 + 8 + 8), psf->sf.frames) ;
  467. psf_binheader_writef (psf, "eh8", data_MARKER16, psf->datalength + 24) ;
  468. psf_fwrite (psf->header, psf->headindex, 1, psf) ;
  469. if (psf->error)
  470. return psf->error ;
  471. psf->dataoffset = psf->headindex ;
  472. if (current > 0)
  473. psf_fseek (psf, current, SEEK_SET) ;
  474. return psf->error ;
  475. } /* w64_write_header */
  476. static int
  477. w64_close (SF_PRIVATE *psf)
  478. {
  479. if (psf->file.mode == SFM_WRITE || psf->file.mode == SFM_RDWR)
  480. w64_write_header (psf, SF_TRUE) ;
  481. return 0 ;
  482. } /* w64_close */