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.

640 lines
16KB

  1. /*
  2. WDL - resample.cpp
  3. Copyright (C) 2010 and later Cockos Incorporated
  4. This software is provided 'as-is', without any express or implied
  5. warranty. In no event will the authors be held liable for any damages
  6. arising from the use of this software.
  7. Permission is granted to anyone to use this software for any purpose,
  8. including commercial applications, and to alter it and redistribute it
  9. freely, subject to the following restrictions:
  10. 1. The origin of this software must not be misrepresented; you must not
  11. claim that you wrote the original software. If you use this software
  12. in a product, an acknowledgment in the product documentation would be
  13. appreciated but is not required.
  14. 2. Altered source versions must be plainly marked as such, and must not be
  15. misrepresented as being the original software.
  16. 3. This notice may not be removed or altered from any source distribution.
  17. You may also distribute this software under the LGPL v2 or later.
  18. */
  19. #include "resample.h"
  20. #include <math.h>
  21. #include "denormal.h"
  22. #ifndef PI
  23. #define PI 3.1415926535897932384626433832795
  24. #endif
  25. class WDL_Resampler::WDL_Resampler_IIRFilter
  26. {
  27. public:
  28. WDL_Resampler_IIRFilter()
  29. {
  30. m_fpos=-1;
  31. Reset();
  32. }
  33. ~WDL_Resampler_IIRFilter()
  34. {
  35. }
  36. void Reset()
  37. {
  38. memset(m_hist,0,sizeof(m_hist));
  39. }
  40. void setParms(double fpos, double Q)
  41. {
  42. if (fabs(fpos-m_fpos)<0.000001) return;
  43. m_fpos=fpos;
  44. double pos = fpos * PI;
  45. double cpos=cos(pos);
  46. double spos=sin(pos);
  47. double alpha=spos/(2.0*Q);
  48. double sc=1.0/( 1 + alpha);
  49. m_b1 = (1-cpos) * sc;
  50. m_b2 = m_b0 = m_b1*0.5;
  51. m_a1 = -2 * cpos * sc;
  52. m_a2 = (1-alpha)*sc;
  53. }
  54. void Apply(WDL_ResampleSample *in1, WDL_ResampleSample *out1, int ns, int span, int w)
  55. {
  56. double b0=m_b0,b1=m_b1,b2=m_b2,a1=m_a1,a2=m_a2;
  57. double *hist=m_hist[w];
  58. while (ns--)
  59. {
  60. double in=*in1;
  61. in1+=span;
  62. double out = (double) ( in*b0 + hist[0]*b1 + hist[1]*b2 - hist[2]*a1 - hist[3]*a2);
  63. hist[1]=hist[0]; hist[0]=in;
  64. hist[3]=hist[2]; *out1 = hist[2]=denormal_filter_double(out);
  65. out1+=span;
  66. }
  67. }
  68. private:
  69. double m_fpos;
  70. double m_a1,m_a2;
  71. double m_b0,m_b1,m_b2;
  72. double m_hist[WDL_RESAMPLE_MAX_FILTERS*WDL_RESAMPLE_MAX_NCH][4];
  73. };
  74. void inline WDL_Resampler::SincSample(WDL_ResampleSample *outptr, const WDL_ResampleSample *inptr, double fracpos, int nch, const WDL_SincFilterSample *filter, int filtsz)
  75. {
  76. const int oversize=m_lp_oversize;
  77. fracpos *= oversize;
  78. const int ifpos=(int)fracpos;
  79. filter += (oversize-ifpos) * filtsz;
  80. fracpos -= ifpos;
  81. int x;
  82. for (x = 0; x < nch; x ++)
  83. {
  84. double sum=0.0,sum2=0.0;
  85. const WDL_SincFilterSample *fptr2=filter;
  86. const WDL_SincFilterSample *fptr=fptr2 - filtsz;
  87. const WDL_ResampleSample *iptr=inptr+x;
  88. int i=filtsz/2;
  89. while (i--)
  90. {
  91. sum += fptr[0]*iptr[0];
  92. sum2 += fptr2[0]*iptr[0];
  93. sum += fptr[1]*iptr[nch];
  94. sum2 += fptr2[1]*iptr[nch];
  95. iptr+=nch*2;
  96. fptr+=2;
  97. fptr2+=2;
  98. }
  99. outptr[x]=sum*fracpos + sum2*(1.0-fracpos);
  100. }
  101. }
  102. void inline WDL_Resampler::SincSample1(WDL_ResampleSample *outptr, const WDL_ResampleSample *inptr, double fracpos, const WDL_SincFilterSample *filter, int filtsz)
  103. {
  104. const int oversize=m_lp_oversize;
  105. fracpos *= oversize;
  106. const int ifpos=(int)fracpos;
  107. fracpos -= ifpos;
  108. double sum=0.0,sum2=0.0;
  109. const WDL_SincFilterSample *fptr2=filter + (oversize-ifpos) * filtsz;
  110. const WDL_SincFilterSample *fptr=fptr2 - filtsz;
  111. const WDL_ResampleSample *iptr=inptr;
  112. int i=filtsz/2;
  113. while (i--)
  114. {
  115. sum += fptr[0]*iptr[0];
  116. sum2 += fptr2[0]*iptr[0];
  117. sum += fptr[1]*iptr[1];
  118. sum2 += fptr2[1]*iptr[1];
  119. iptr+=2;
  120. fptr+=2;
  121. fptr2+=2;
  122. }
  123. outptr[0]=sum*fracpos+sum2*(1.0-fracpos);
  124. }
  125. void inline WDL_Resampler::SincSample2(WDL_ResampleSample *outptr, const WDL_ResampleSample *inptr, double fracpos, const WDL_SincFilterSample *filter, int filtsz)
  126. {
  127. const int oversize=m_lp_oversize;
  128. fracpos *= oversize;
  129. const int ifpos=(int)fracpos;
  130. fracpos -= ifpos;
  131. const WDL_SincFilterSample *fptr2=filter + (oversize-ifpos) * filtsz;
  132. const WDL_SincFilterSample *fptr=fptr2 - filtsz;
  133. double sum=0.0;
  134. double sum2=0.0;
  135. double sumb=0.0;
  136. double sum2b=0.0;
  137. const WDL_ResampleSample *iptr=inptr;
  138. int i=filtsz/2;
  139. while (i--)
  140. {
  141. sum += fptr[0]*iptr[0];
  142. sum2 += fptr[0]*iptr[1];
  143. sumb += fptr2[0]*iptr[0];
  144. sum2b += fptr2[0]*iptr[1];
  145. sum += fptr[1]*iptr[2];
  146. sum2 += fptr[1]*iptr[3];
  147. sumb += fptr2[1]*iptr[2];
  148. sum2b += fptr2[1]*iptr[3];
  149. iptr+=4;
  150. fptr+=2;
  151. fptr2+=2;
  152. }
  153. outptr[0]=sum*fracpos + sumb*(1.0-fracpos);
  154. outptr[1]=sum2*fracpos + sum2b*(1.0-fracpos);
  155. }
  156. WDL_Resampler::WDL_Resampler()
  157. {
  158. m_filterq=0.707f;
  159. m_filterpos=0.693f; // .792 ?
  160. m_sincoversize=0;
  161. m_lp_oversize=1;
  162. m_sincsize=0;
  163. m_filtercnt=1;
  164. m_interp=true;
  165. m_feedmode=false;
  166. m_filter_coeffs_size=0;
  167. m_sratein=44100.0;
  168. m_srateout=44100.0;
  169. m_ratio=1.0;
  170. m_filter_ratio=-1.0;
  171. m_iirfilter=0;
  172. Reset();
  173. }
  174. WDL_Resampler::~WDL_Resampler()
  175. {
  176. delete m_iirfilter;
  177. }
  178. void WDL_Resampler::Reset(double fracpos)
  179. {
  180. m_last_requested=0;
  181. m_filtlatency=0;
  182. m_fracpos=fracpos;
  183. m_samples_in_rsinbuf=0;
  184. if (m_iirfilter) m_iirfilter->Reset();
  185. }
  186. void WDL_Resampler::SetMode(bool interp, int filtercnt, bool sinc, int sinc_size, int sinc_interpsize)
  187. {
  188. m_sincsize = sinc && sinc_size>= 4 ? sinc_size > 8192 ? 8192 : (sinc_size&~1) : 0;
  189. m_sincoversize = m_sincsize ? (sinc_interpsize<= 1 ? 1 : sinc_interpsize>=8192 ? 8192 : sinc_interpsize) : 1;
  190. m_filtercnt = m_sincsize ? 0 : (filtercnt<=0?0 : filtercnt >= WDL_RESAMPLE_MAX_FILTERS ? WDL_RESAMPLE_MAX_FILTERS : filtercnt);
  191. m_interp=interp && !m_sincsize;
  192. // char buf[512];
  193. // sprintf(buf,"setting interp=%d, filtercnt=%d, sinc=%d,%d\n",m_interp,m_filtercnt,m_sincsize,m_sincoversize);
  194. // OutputDebugString(buf);
  195. if (!m_sincsize)
  196. {
  197. m_filter_coeffs.Resize(0);
  198. m_filter_coeffs_size=0;
  199. }
  200. if (!m_filtercnt)
  201. {
  202. delete m_iirfilter;
  203. m_iirfilter=0;
  204. }
  205. }
  206. void WDL_Resampler::SetRates(double rate_in, double rate_out)
  207. {
  208. if (rate_in<1.0) rate_in=1.0;
  209. if (rate_out<1.0) rate_out=1.0;
  210. if (rate_in != m_sratein || rate_out != m_srateout)
  211. {
  212. m_sratein=rate_in;
  213. m_srateout=rate_out;
  214. m_ratio=m_sratein / m_srateout;
  215. }
  216. }
  217. void WDL_Resampler::BuildLowPass(double filtpos) // only called in sinc modes
  218. {
  219. const int wantsize=m_sincsize;
  220. const int wantinterp=m_sincoversize;
  221. if (m_filter_ratio!=filtpos ||
  222. m_filter_coeffs_size != wantsize ||
  223. m_lp_oversize != wantinterp)
  224. {
  225. m_lp_oversize = wantinterp;
  226. m_filter_ratio=filtpos;
  227. // build lowpass filter
  228. const int allocsize = wantsize*(m_lp_oversize+1);
  229. WDL_SincFilterSample *cfout=m_filter_coeffs.Resize(allocsize);
  230. if (m_filter_coeffs.GetSize()==allocsize)
  231. {
  232. m_filter_coeffs_size=wantsize;
  233. const double dwindowpos = 2.0 * PI/(double)wantsize;
  234. const double dsincpos = PI * filtpos; // filtpos is outrate/inrate, i.e. 0.5 is going to half rate
  235. const int hwantsize=wantsize/2;
  236. double filtpower=0.0;
  237. WDL_SincFilterSample *ptrout = cfout;
  238. int slice;
  239. for (slice=0;slice<=wantinterp;slice++)
  240. {
  241. const double frac = slice / (double)wantinterp;
  242. const int center_x = slice == 0 ? hwantsize : slice == wantinterp ? hwantsize-1 : -1;
  243. int x;
  244. for (x=0;x<wantsize;x++)
  245. {
  246. if (x==center_x)
  247. {
  248. // we know this will be 1.0
  249. *ptrout++ = 1.0;
  250. }
  251. else
  252. {
  253. const double xfrac = frac + x;
  254. const double windowpos = dwindowpos * xfrac;
  255. const double sincpos = dsincpos * (xfrac - hwantsize);
  256. // blackman-harris * sinc
  257. const double val = (0.35875 - 0.48829 * cos(windowpos) + 0.14128 * cos(2*windowpos) - 0.01168 * cos(3*windowpos)) * sin(sincpos) / sincpos;
  258. if (slice<wantinterp) filtpower+=val;
  259. *ptrout++ = (WDL_SincFilterSample)val;
  260. }
  261. }
  262. }
  263. filtpower = wantinterp/(filtpower+1.0);
  264. int x;
  265. for (x = 0; x < allocsize; x ++)
  266. {
  267. cfout[x] = (WDL_SincFilterSample) (cfout[x]*filtpower);
  268. }
  269. }
  270. else m_filter_coeffs_size=0;
  271. }
  272. }
  273. double WDL_Resampler::GetCurrentLatency()
  274. {
  275. double v=((double)m_samples_in_rsinbuf-m_filtlatency)/m_sratein;
  276. if (v<0.0)v=0.0;
  277. return v;
  278. }
  279. int WDL_Resampler::ResamplePrepare(int out_samples, int nch, WDL_ResampleSample **inbuffer)
  280. {
  281. if (nch > WDL_RESAMPLE_MAX_NCH || nch < 1) return 0;
  282. int fsize=0;
  283. if (m_sincsize>1) fsize = m_sincsize;
  284. int hfs=fsize/2;
  285. if (hfs>1 && m_samples_in_rsinbuf<hfs-1)
  286. {
  287. m_filtlatency+=hfs-1 - m_samples_in_rsinbuf;
  288. m_samples_in_rsinbuf=hfs-1;
  289. if (m_samples_in_rsinbuf>0)
  290. {
  291. WDL_ResampleSample *p = m_rsinbuf.Resize(m_samples_in_rsinbuf*nch,false);
  292. memset(p,0,sizeof(WDL_ResampleSample)*m_rsinbuf.GetSize());
  293. }
  294. }
  295. int sreq = 0;
  296. if (!m_feedmode) sreq = (int)(m_ratio * out_samples) + 4 + fsize - m_samples_in_rsinbuf;
  297. else sreq = out_samples;
  298. if (sreq<0)sreq=0;
  299. again:
  300. m_rsinbuf.Resize((m_samples_in_rsinbuf+sreq)*nch,false);
  301. int sz = m_rsinbuf.GetSize()/(nch?nch:1) - m_samples_in_rsinbuf;
  302. if (sz!=sreq)
  303. {
  304. if (sreq>4 && !sz)
  305. {
  306. sreq/=2;
  307. goto again; // try again with half the size
  308. }
  309. // todo: notify of error?
  310. sreq=sz;
  311. }
  312. *inbuffer = m_rsinbuf.Get() + m_samples_in_rsinbuf*nch;
  313. m_last_requested=sreq;
  314. return sreq;
  315. }
  316. int WDL_Resampler::ResampleOut(WDL_ResampleSample *out, int nsamples_in, int nsamples_out, int nch)
  317. {
  318. if (nch > WDL_RESAMPLE_MAX_NCH || nch < 1) return 0;
  319. if (m_filtercnt>0)
  320. {
  321. if (m_ratio > 1.0 && nsamples_in > 0) // filter input
  322. {
  323. if (!m_iirfilter) m_iirfilter = new WDL_Resampler_IIRFilter;
  324. int n=m_filtercnt;
  325. m_iirfilter->setParms((1.0/m_ratio)*m_filterpos,m_filterq);
  326. WDL_ResampleSample *buf=(WDL_ResampleSample *)m_rsinbuf.Get() + m_samples_in_rsinbuf*nch;
  327. int a,x;
  328. int offs=0;
  329. for (x=0; x < nch; x ++)
  330. for (a = 0; a < n; a ++)
  331. m_iirfilter->Apply(buf+x,buf+x,nsamples_in,nch,offs++);
  332. }
  333. }
  334. // prevent the caller from corrupting the internal state
  335. m_samples_in_rsinbuf += nsamples_in < m_last_requested ? nsamples_in : m_last_requested;
  336. int rsinbuf_availtemp = m_samples_in_rsinbuf;
  337. if (nsamples_in < m_last_requested) // flush out to ensure we can deliver
  338. {
  339. int fsize=(m_last_requested-nsamples_in)*2 + m_sincsize*2;
  340. int alloc_size=(m_samples_in_rsinbuf+fsize)*nch;
  341. WDL_ResampleSample *zb=m_rsinbuf.Resize(alloc_size,false);
  342. if (m_rsinbuf.GetSize()==alloc_size)
  343. {
  344. memset(zb+m_samples_in_rsinbuf*nch,0,fsize*nch*sizeof(WDL_ResampleSample));
  345. rsinbuf_availtemp = m_samples_in_rsinbuf+fsize;
  346. }
  347. }
  348. int ret=0;
  349. double srcpos=m_fracpos;
  350. double drspos = m_ratio;
  351. WDL_ResampleSample *localin = m_rsinbuf.Get();
  352. WDL_ResampleSample *outptr=out;
  353. int ns=nsamples_out;
  354. int outlatadj=0;
  355. if (m_sincsize) // sinc interpolating
  356. {
  357. if (m_ratio > 1.0) BuildLowPass(1.0 / (m_ratio*1.03));
  358. else BuildLowPass(1.0);
  359. int filtsz=m_filter_coeffs_size;
  360. int filtlen = rsinbuf_availtemp - filtsz;
  361. outlatadj=filtsz/2-1;
  362. WDL_SincFilterSample *filter=m_filter_coeffs.Get();
  363. if (nch == 1)
  364. {
  365. while (ns--)
  366. {
  367. int ipos = (int)srcpos;
  368. if (ipos >= filtlen-1) break; // quit decoding, not enough input samples
  369. SincSample1(outptr,localin + ipos,srcpos-ipos,filter,filtsz);
  370. outptr ++;
  371. srcpos+=drspos;
  372. ret++;
  373. }
  374. }
  375. else if (nch==2)
  376. {
  377. while (ns--)
  378. {
  379. int ipos = (int)srcpos;
  380. if (ipos >= filtlen-1) break; // quit decoding, not enough input samples
  381. SincSample2(outptr,localin + ipos*2,srcpos-ipos,filter,filtsz);
  382. outptr+=2;
  383. srcpos+=drspos;
  384. ret++;
  385. }
  386. }
  387. else
  388. {
  389. while (ns--)
  390. {
  391. int ipos = (int)srcpos;
  392. if (ipos >= filtlen-1) break; // quit decoding, not enough input samples
  393. SincSample(outptr,localin + ipos*nch,srcpos-ipos,nch,filter,filtsz);
  394. outptr += nch;
  395. srcpos+=drspos;
  396. ret++;
  397. }
  398. }
  399. }
  400. else if (!m_interp) // point sampling
  401. {
  402. if (nch == 1)
  403. {
  404. while (ns--)
  405. {
  406. int ipos = (int)srcpos;
  407. if (ipos >= rsinbuf_availtemp) break; // quit decoding, not enough input samples
  408. *outptr++ = localin[ipos];
  409. srcpos+=drspos;
  410. ret++;
  411. }
  412. }
  413. else if (nch == 2)
  414. {
  415. while (ns--)
  416. {
  417. int ipos = (int)srcpos;
  418. if (ipos >= rsinbuf_availtemp) break; // quit decoding, not enough input samples
  419. ipos+=ipos;
  420. outptr[0] = localin[ipos];
  421. outptr[1] = localin[ipos+1];
  422. outptr+=2;
  423. srcpos+=drspos;
  424. ret++;
  425. }
  426. }
  427. else
  428. while (ns--)
  429. {
  430. int ipos = (int)srcpos;
  431. if (ipos >= rsinbuf_availtemp) break; // quit decoding, not enough input samples
  432. memcpy(outptr,localin + ipos*nch,nch*sizeof(WDL_ResampleSample));
  433. outptr += nch;
  434. srcpos+=drspos;
  435. ret++;
  436. }
  437. }
  438. else // linear interpolation
  439. {
  440. if (nch == 1)
  441. {
  442. while (ns--)
  443. {
  444. int ipos = (int)srcpos;
  445. double fracpos=srcpos-ipos;
  446. if (ipos >= rsinbuf_availtemp-1)
  447. {
  448. break; // quit decoding, not enough input samples
  449. }
  450. double ifracpos=1.0-fracpos;
  451. WDL_ResampleSample *inptr = localin + ipos;
  452. *outptr++ = inptr[0]*(ifracpos) + inptr[1]*(fracpos);
  453. srcpos+=drspos;
  454. ret++;
  455. }
  456. }
  457. else if (nch == 2)
  458. {
  459. while (ns--)
  460. {
  461. int ipos = (int)srcpos;
  462. double fracpos=srcpos-ipos;
  463. if (ipos >= rsinbuf_availtemp-1)
  464. {
  465. break; // quit decoding, not enough input samples
  466. }
  467. double ifracpos=1.0-fracpos;
  468. WDL_ResampleSample *inptr = localin + ipos*2;
  469. outptr[0] = inptr[0]*(ifracpos) + inptr[2]*(fracpos);
  470. outptr[1] = inptr[1]*(ifracpos) + inptr[3]*(fracpos);
  471. outptr += 2;
  472. srcpos+=drspos;
  473. ret++;
  474. }
  475. }
  476. else
  477. {
  478. while (ns--)
  479. {
  480. int ipos = (int)srcpos;
  481. double fracpos=srcpos-ipos;
  482. if (ipos >= rsinbuf_availtemp-1)
  483. {
  484. break; // quit decoding, not enough input samples
  485. }
  486. double ifracpos=1.0-fracpos;
  487. int ch=nch;
  488. WDL_ResampleSample *inptr = localin + ipos*nch;
  489. while (ch--)
  490. {
  491. *outptr++ = inptr[0]*(ifracpos) + inptr[nch]*(fracpos);
  492. inptr++;
  493. }
  494. srcpos+=drspos;
  495. ret++;
  496. }
  497. }
  498. }
  499. if (m_filtercnt>0)
  500. {
  501. if (m_ratio < 1.0 && ret>0) // filter output
  502. {
  503. if (!m_iirfilter) m_iirfilter = new WDL_Resampler_IIRFilter;
  504. int n=m_filtercnt;
  505. m_iirfilter->setParms(m_ratio*m_filterpos,m_filterq);
  506. int x,a;
  507. int offs=0;
  508. for (x=0; x < nch; x ++)
  509. for (a = 0; a < n; a ++)
  510. m_iirfilter->Apply(out+x,out+x,ret,nch,offs++);
  511. }
  512. }
  513. if (ret>0 && rsinbuf_availtemp>m_samples_in_rsinbuf) // we had to pad!!
  514. {
  515. // check for the case where rsinbuf_availtemp>m_samples_in_rsinbuf, decrease ret down to actual valid samples
  516. double adj=(srcpos-m_samples_in_rsinbuf + outlatadj) / drspos;
  517. if (adj>0)
  518. {
  519. ret -= (int) (adj + 0.5);
  520. if (ret<0)ret=0;
  521. }
  522. }
  523. int isrcpos=(int)srcpos;
  524. if (isrcpos > m_samples_in_rsinbuf) isrcpos=m_samples_in_rsinbuf;
  525. m_fracpos = srcpos - isrcpos;
  526. m_samples_in_rsinbuf -= isrcpos;
  527. if (m_samples_in_rsinbuf <= 0) m_samples_in_rsinbuf=0;
  528. else
  529. memmove(localin, localin + isrcpos*nch,m_samples_in_rsinbuf*sizeof(WDL_ResampleSample)*nch);
  530. return ret;
  531. }