Leaked source code of windows server 2003
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.

4288 lines
132 KiB

  1. /********************************************************/
  2. /* nabtsdsp.c */
  3. /* */
  4. /* Perform DSP required for NABTS decoding */
  5. /* */
  6. /* Copyright Microsoft, Inc. 1997. All rights reserved. */
  7. /********************************************************/
  8. #include <stdio.h>
  9. #include <ctype.h>
  10. #include <stdlib.h>
  11. #include <memory.h>
  12. #include <assert.h>
  13. #ifdef NTOOL
  14. # include <math.h>
  15. # include "winsys.h"
  16. #else //NTOOL
  17. # include <strmini.h>
  18. # include <ksmedia.h>
  19. # include <kskludge.h>
  20. #endif //!NTOOL
  21. #include "host.h"
  22. #include "nabtsapi.h"
  23. #include "nabtsprv.h"
  24. #include "nabtsdsp.h"
  25. #pragma warning(disable:4244 4305)
  26. inline int fltCmp(Double a, Double b) { return (a == b); }
  27. #define NDSP_STATE_MAGIC 0x648a0a59
  28. /***************/
  29. /*** Globals ***/
  30. /***************/
  31. /* Have the global GCR signals been initialized?
  32. 0 if no, 1 if yes.
  33. See NDSPInitGlobals */
  34. int g_bNDSPGlobalsInitted= 0;
  35. /* How many fields to check during retrain operation */
  36. int g_nNabtsRetrainDuration= 16; /* in fields */
  37. /* How many fields between automatic retrains.
  38. (Note that immediate retrain can be caused also by
  39. NDSPStartRetrain) */
  40. int g_nNabtsRetrainDelay= 600; /* in fields */
  41. /* How many "far ghost" equalization taps should we add?
  42. Note that this only occurs when we are using the GCR for
  43. equalization/ghost cancellation. The NABTS sync is not long
  44. enough to be able to detect and cancel far ghosts. */
  45. int g_nNabtsAdditionalTapsGCR= 2; /* # of additional taps to add while equalizing */
  46. /* The following is extraneous and should be removed at earliest
  47. convenience: */
  48. Double g_dGCRDetectionThreshold= 100000.0; /* extraneous */
  49. /* Minimum and maximum limits to the sample # at which the first NABTS
  50. sync sample occurs.
  51. These are used by NDSPDecodeFindSync.
  52. Be careful changing these numbers, as you want to make sure that
  53. there are enough samples left at the beginning and end of the lines
  54. to do adequate filtering.
  55. As of the time of writing these comments, the range is 20 to 85.
  56. This means that VBI signal should be adjusted such that the
  57. GCR begins roughly at (20+85)/2, or around sample 52.
  58. To verify adjustment of the VBI signal, enable the
  59. "Sync found at %d with conv ..." debugging line in NDSPDecodeLine
  60. and plot a histogram of the found sync positions.
  61. Maladjustment of the VBI signal probably means incorrect values in the
  62. VBIINFOHEADER structure passed with the signal.
  63. */
  64. /* replace these w/ time-based values... */
  65. //int g_nNabtsDecodeBeginMin= 20;
  66. /* BeginMax should be less than
  67. DecodeSamples - (36*8*5) - 10 */
  68. //int g_nNabtsDecodeBeginMax= 85;
  69. int g_nNabtsDecodeBeginStart;
  70. int g_nNabtsDecodeBeginEnd;
  71. /* What's the longest post-ghost which we can correct without requiring
  72. samples we don't have? */
  73. int g_nNabtsMaxPostGhost= 170; /* in samples */
  74. /* What's the longest pre-ghost which we can correct without requiring
  75. samples we don't have? */
  76. int g_nNabtsMaxPreGhost= 40; /* in samples */
  77. /* This is the default filter used for signals if both GCR-based and
  78. NABTS sync-based equalization/ghost cancellation algorithms fail */
  79. FIRFilter g_filterDefault5= {
  80. FALSE, /* bVariableTaps */
  81. 5, /* nTaps */
  82. 5, /* dTapSpacing */
  83. 0, /* nMinTap */
  84. 0, /* nMaxTap */
  85. { 0.0549969,
  86. -0.115282,
  87. 0.946446,
  88. -0.260854,
  89. 0.0476174}, /* pdTaps */
  90. //{} /* pnTapLocs */
  91. };
  92. FIRFilter g_filterDefault47 = {
  93. FALSE, /* bVariableTaps */
  94. 5, /* nTaps */
  95. KS_47NABTS_SCALER, /* dTapSpacing */ //FIXME (maybe)
  96. 0, /* nMinTap */
  97. 0, /* nMaxTap */
  98. { 0.0549969,
  99. -0.115282,
  100. 0.946446,
  101. -0.260854,
  102. 0.0476174}, /* pdTaps */
  103. //{} /* pnTapLocs */
  104. };
  105. FIRFilter g_filterDefault4= {
  106. FALSE, /* bVariableTaps */
  107. 5, /* nTaps */
  108. 4, /* dTapSpacing */
  109. 0, /* nMinTap */
  110. 0, /* nMaxTap */
  111. { 0.0549969,
  112. -0.115282,
  113. 0.946446,
  114. -0.260854,
  115. 0.0476174}, /* pdTaps */
  116. //{} /* pnTapLocs */
  117. };
  118. FIRFilter* g_filterDefault = &g_filterDefault5;
  119. /* If DEBUG_VERBOSE, do we show information about NABTS sync? */
  120. int g_bDebugSyncBytes= FALSE;
  121. /* This is a set of samples representing a typical NABTS sync taken at
  122. NABTS bit rate * 5.
  123. It is used for NABTS sync equalization, which is a fallback from the
  124. GCR equalization/ghost cancellation
  125. */
  126. Double g_pdSync5[NABSYNC_SIZE]={
  127. 72.7816,
  128. 98.7424,
  129. 128.392,
  130. 152.351,
  131. 161.749,
  132. 153.177,
  133. 127.707,
  134. 94.7642,
  135. 67.3989,
  136. 57.3863,
  137. 68.85,
  138. 97.1427,
  139. 131.288,
  140. 156.581,
  141. 164.365,
  142. 151.438,
  143. 123.222,
  144. 90.791,
  145. 66.191,
  146. 57.8995,
  147. 70.1041,
  148. 98.0289,
  149. 129.88,
  150. 154.471,
  151. 161.947,
  152. 149.796,
  153. 123.056,
  154. 90.4673,
  155. 64.7314,
  156. 55.3073,
  157. 67.6522,
  158. 96.0781,
  159. 129.079,
  160. 154.704,
  161. 162.274,
  162. 148.879,
  163. 120.957,
  164. 88.971,
  165. 64.2987,
  166. 57.3871,
  167. 69.9341,
  168. 97.2937,
  169. 130.365,
  170. 154.517,
  171. 161.46,
  172. 149.066,
  173. 122.341,
  174. 89.6801,
  175. 64.8201,
  176. 56.3087,
  177. 67.9328,
  178. 95.4871,
  179. 127.708,
  180. 152.344,
  181. 160.203,
  182. 147.785,
  183. 120.972,
  184. 89.2391,
  185. 64.8204,
  186. 55.4777,
  187. 67.4764,
  188. 95.4465,
  189. 128.277,
  190. 153.469,
  191. 161.047,
  192. 149.581,
  193. 123.23,
  194. 90.3241,
  195. 65.0892,
  196. 57.5301,
  197. 69.3996,
  198. 97.4484,
  199. 130.064,
  200. 154.063,
  201. 161.211,
  202. 147.833,
  203. 119.759,
  204. 89.3406,
  205. 67.3504,
  206. 60.1918,
  207. 70.7763,
  208. 93.4328,
  209. 120.709,
  210. 145.109,
  211. 160.585,
  212. 166.873,
  213. 166.863,
  214. 163.397,
  215. 159.525,
  216. 158.707,
  217. 160.963,
  218. 165.351,
  219. 167.585,
  220. 165.082,
  221. 154.448,
  222. 135.721,
  223. 112.454,
  224. 88.9855,
  225. 68.8211,
  226. 55.3262,
  227. 47.8155,
  228. 45.8446,
  229. 47.352,
  230. 51.1301,
  231. 59.5515,
  232. 74.6295,
  233. 96.5327,
  234. 121.793,
  235. 144.703,
  236. 160.891,
  237. 167.617,
  238. 164.946,
  239. 159.698,
  240. 156.271,
  241. 157.871
  242. };
  243. Double g_pdSync47[NABSYNC_SIZE] = {
  244. 72.7816,
  245. 96.9455,
  246. 125.488,
  247. 150.642,
  248. 163.827,
  249. 160.895,
  250. 139.686,
  251. 106.374,
  252. 72.2535,
  253. 51.1334,
  254. 51.7029,
  255. 74.3792,
  256. 112.893,
  257. 150.448,
  258. 175.333,
  259. 177.089,
  260. 154.67,
  261. 66.4422,
  262. 56.79,
  263. 65.8731,
  264. 91.2727,
  265. 123.173,
  266. 151.979,
  267. 166.734,
  268. 161.95,
  269. 139.844,
  270. 105.285,
  271. 70.7285,
  272. 46.7033,
  273. 46.1175,
  274. 69.0775,
  275. 106.56,
  276. 147.593,
  277. 175.669,
  278. 122.895,
  279. 91.9615,
  280. 65.5553,
  281. 54.3455,
  282. 61.6435,
  283. 85.2679,
  284. 120.119,
  285. 151.151,
  286. 168.22,
  287. 165.263,
  288. 144.115,
  289. 107.76,
  290. 71.526,
  291. 46.4459,
  292. 42.8836,
  293. 64.2428,
  294. 152.106,
  295. 161.332,
  296. 151.847,
  297. 127.703,
  298. 95.8986,
  299. 67.9346,
  300. 50.751,
  301. 54.7629,
  302. 78.5341,
  303. 113.773,
  304. 148.647,
  305. 169.038,
  306. 169.544,
  307. 150.153,
  308. 112.5,
  309. 72.1901,
  310. 45.6607,
  311. 95.472,
  312. 127.155,
  313. 152.763,
  314. 164.454,
  315. 156.34,
  316. 130.82,
  317. 98.6696,
  318. 70.8212,
  319. 54.4185,
  320. 57.0453,
  321. 75.2489,
  322. 102.964,
  323. 132.916,
  324. 155.25,
  325. 166.882,
  326. 170.224,
  327. 159.55,
  328. 158.502,
  329. 160.298,
  330. 164.877,
  331. 168.268,
  332. 168.627,
  333. 161.825,
  334. 146.297,
  335. 124.544,
  336. 100.595,
  337. 77.4086,
  338. 60.5608,
  339. 49.3086,
  340. 44.6113,
  341. 44.0319,
  342. 43.2192,
  343. 44.4737,
  344. 95.0021,
  345. 119.016,
  346. 141.76,
  347. 159.261,
  348. 168.426,
  349. 166.854,
  350. 161.152,
  351. 155.495,
  352. };
  353. Double g_pdSync4[NABSYNC_SIZE] = {
  354. -39.1488,
  355. -12.0978,
  356. 33.5648,
  357. 46.2761,
  358. 39.4813,
  359. 9.29926,
  360. -35.5121,
  361. -54.1576,
  362. -43.0804,
  363. -8.89138,
  364. 34.4705,
  365. 51.1527,
  366. 37.7423,
  367. 3.70292,
  368. -37.0844,
  369. -54.5635,
  370. -41.8263,
  371. -8.01077,
  372. 31.8198,
  373. 49.2458,
  374. 36.1003,
  375. 3.28592,
  376. -37.4172,
  377. -57.2909,
  378. -44.2782,
  379. -9.47973,
  380. 31.2838,
  381. 49.8912,
  382. 35.1833,
  383. 1.00765,
  384. -38.2074,
  385. -56.0426,
  386. -41.9963,
  387. -8.0181,
  388. 31.6699,
  389. 49.2505,
  390. 35.3703,
  391. 2.16505,
  392. -37.3994,
  393. -56.7359,
  394. -43.9976,
  395. -9.86633,
  396. 28.8379,
  397. 47.9894,
  398. 34.0893,
  399. 0.793831,
  400. -37.1945,
  401. -57.5387,
  402. -44.454,
  403. -9.71416,
  404. 29.6742,
  405. 48.954,
  406. 35.8853,
  407. 2.77469,
  408. -36.6699,
  409. -56.0604,
  410. -42.5308,
  411. -7.411,
  412. 30.524,
  413. 49.4549,
  414. 34.1373,
  415. -0.682732,
  416. -35.4269,
  417. -53.5466,
  418. -41.1541,
  419. -12.8922,
  420. 20.253,
  421. 45.9943,
  422. 53.1773,
  423. 53.4395,
  424. 49.3341,
  425. 44.4674,
  426. 49.0326,
  427. 53.631,
  428. 53.0627,
  429. 46.721,
  430. 22.0253,
  431. -6.8225,
  432. -32.9031,
  433. -56.6665,
  434. -64.1149,
  435. -65.8565,
  436. -66.4073,
  437. -53.3933,
  438. -39.0662,
  439. -11.9141,
  440. 25.1485,
  441. 40.7185,
  442. 55.6866,
  443. 56.0335,
  444. 31.9488,
  445. 64.5845,
  446. -112.813
  447. };
  448. int g_nNabtsSyncSize = NABSYNC_SIZE;
  449. Double g_pdSync[MAX_NABTS_SAMPLES_PER_LINE]; /* used for resampling */
  450. /* This is a set of samples representing a typical GCR waveform taken
  451. at NABTS bit rate * 2.5.
  452. It is used for GCR-based equalization and ghost cancellation, which
  453. is the primary and most desirable strategy
  454. */
  455. Double g_pdGCRSignal1_5[GCR_SIZE]={
  456. 0.00000000000000E+0000, 1.02260999999970E-0003, 1.05892999999924E-0003,
  457. 1.09718000000036E-0003, 1.13701999999982E-0003, 1.17906999999917E-0003,
  458. 1.22286999999943E-0003, 1.26910000000002E-0003, 1.31748000000087E-0003,
  459. 1.36846999999918E-0003, 1.42190000000042E-0003, 1.47821000000015E-0003,
  460. 1.53740000000013E-0003, 1.59988999999960E-0003, 1.66547000000072E-0003,
  461. 1.73485999999912E-0003, 1.80787000000038E-0003, 1.88510000000086E-0003,
  462. 1.96657000000044E-0003, 2.05283000000023E-0003, 2.14390000000009E-0003,
  463. 2.24045999999944E-0003, 2.34250000000102E-0003, 2.45085999999972E-0003,
  464. 2.56557000000157E-0003, 2.68752999999933E-0003, 2.81685999999937E-0003,
  465. 2.95447999999965E-0003, 3.10071999999906E-0003, 3.25656000000052E-0003,
  466. 3.42231000000126E-0003, 3.59925000000061E-0003, 3.78781999999944E-0003,
  467. 3.98931000000147E-0003, 4.20439999999900E-0003, 4.43461999999784E-0003,
  468. 4.68077000000022E-0003, 4.94463999999795E-0003, 5.22724000000352E-0003,
  469. 5.53064000000347E-0003, 5.85610000000258E-0003, 6.20609999999999E-0003,
  470. 6.58219999999687E-0003, 6.98715000000050E-0003, 7.42307000000153E-0003,
  471. 7.89319000000432E-0003, 8.40024000000028E-0003, 8.94780000000139E-0003,
  472. 9.53912999999318E-0003, 1.01788000000056E-0002, 1.08706000000041E-0002,
  473. 1.16199999999935E-0002, 1.24318000000017E-0002, 1.33122999999955E-0002,
  474. 1.42673999999943E-0002, 1.53047000000015E-0002, 1.64313999999877E-0002,
  475. 1.76562000000047E-0002, 1.89881000000014E-0002, 2.04373999999916E-0002,
  476. 2.20148999999878E-0002, 2.37329999999929E-0002, 2.56043999999918E-0002,
  477. 2.76438999999868E-0002, 2.98664999999971E-0002, 3.22896999999784E-0002,
  478. 3.49309999999718E-0002, 3.78107999999884E-0002, 4.09500000000094E-0002,
  479. 4.43718999999874E-0002, 4.81004000000098E-0002, 5.21621999999979E-0002,
  480. 5.65846999999735E-0002, 6.13976000000207E-0002, 6.66313000000400E-0002,
  481. 7.23186999999825E-0002, 7.84923999999592E-0002, 8.51879000000508E-0002,
  482. 9.24387000000024E-0002, 1.00280999999995E-0001, 1.08748999999989E-0001,
  483. 1.17876000000024E-0001, 1.27694000000020E-0001, 1.38230999999905E-0001,
  484. 1.49511000000075E-0001, 1.61552999999913E-0001, 1.74367999999959E-0001,
  485. 1.87959000000092E-0001, 2.02316999999994E-0001, 2.17419999999947E-0001,
  486. 2.33228000000054E-0001, 2.49684999999999E-0001, 2.66712000000098E-0001,
  487. 2.84204999999929E-0001, 3.02032999999938E-0001, 3.20032000000083E-0001,
  488. 3.38005000000067E-0001, 3.55717999999797E-0001, 3.72897000000194E-0001,
  489. 3.89230000000225E-0001, 4.04360999999881E-0001, 4.17892999999822E-0001,
  490. 4.29389999999785E-0001, 4.38381999999820E-0001, 4.44367000000057E-0001,
  491. 4.46823000000222E-0001, 4.45215000000189E-0001, 4.39011999999821E-0001,
  492. 4.27705000000060E-0001, 4.10821999999825E-0001, 3.87955000000147E-0001,
  493. 3.58785999999782E-0001, 3.23113999999805E-0001, 2.80886000000010E-0001,
  494. 2.32228000000077E-0001, 1.77474000000075E-0001, 1.17197000000033E-0001,
  495. 5.22252000000094E-0002, -1.63339000000065E-0002, -8.70900999999549E-0002,
  496. -1.58376000000089E-0001, -2.28262000000086E-0001, -2.94597000000067E-0001,
  497. -3.55059999999867E-0001, -4.07236000000012E-0001, -4.48710000000119E-0001,
  498. -4.77181000000201E-0001, -4.90596000000096E-0001, -4.87282999999934E-0001,
  499. -4.66113999999834E-0001, -4.26633000000038E-0001, -3.69197999999869E-0001,
  500. -2.95083999999861E-0001, -2.06552999999985E-0001, -1.06861999999978E-0001,
  501. -2.26569000000065E-0004, 1.08304999999973E-0001, 2.13048999999955E-0001,
  502. 3.07967000000190E-0001, 3.87016000000131E-0001, 4.44562000000133E-0001,
  503. 4.75829999999860E-0001, 4.77361000000201E-0001, 4.47440999999799E-0001,
  504. 3.86449999999968E-0001, 2.97098000000005E-0001, 1.84488000000101E-0001,
  505. 5.59898000000203E-0002, -7.91186000000152E-0002, -2.10235999999895E-0001,
  506. -3.26232999999775E-0001, -4.16419000000133E-0001, -4.71591999999873E-0001,
  507. -4.85105000000203E-0001, -4.53794000000016E-0001, -3.78679999999804E-0001,
  508. -2.65297999999802E-0001, -1.23576999999955E-0001, 3.28038999999762E-0002,
  509. 1.87584000000015E-0001, 3.23534999999993E-0001, 4.24414000000070E-0001,
  510. 4.77027999999791E-0001, 4.73159000000123E-0001, 4.11086000000068E-0001,
  511. 2.96408000000156E-0001, 1.41969000000017E-0001, -3.32369999999855E-0002,
  512. -2.06171999999924E-0001, -3.52687000000060E-0001, -4.50949999999921E-0001,
  513. -4.84910000000127E-0001, -4.47235999999975E-0001, -3.41190000000097E-0001,
  514. -1.80984000000080E-0001, 9.63199000000259E-0003, 2.00563999999986E-0001,
  515. 3.59968000000208E-0001, 4.59641999999803E-0001, 4.80308000000150E-0001,
  516. 4.15771999999833E-0001, 2.75076999999783E-0001, 8.19416000000501E-0002,
  517. -1.28693999999996E-0001, -3.16683999999896E-0001, -4.44307000000208E-0001,
  518. -4.84112000000096E-0001, -4.25353000000086E-0001, -2.77427999999873E-0001,
  519. -6.92708000000266E-0002, 1.55655000000024E-0001, 3.48071999999775E-0001,
  520. 4.63710999999876E-0001, 4.73907999999938E-0001, 3.73398000000179E-0001,
  521. 1.83113999999932E-0001, -5.32258000000070E-0002, -2.78533000000152E-0001,
  522. -4.35938999999962E-0001, -4.83456999999817E-0001, -4.05898999999863E-0001,
  523. -2.20585999999912E-0001, 2.53643000000068E-0002, 2.66195000000153E-0001,
  524. 4.34799999999996E-0001, 4.81733999999960E-0001, 3.90551999999843E-0001,
  525. 1.84567999999899E-0001, -7.78073999999833E-0002, -3.18760000000111E-0001,
  526. -4.63948000000073E-0001, -4.65948999999910E-0001, -3.20834999999988E-0001,
  527. -7.18414000000394E-0002, 2.02201999999943E-0001, 4.11149000000023E-0001,
  528. 4.83345000000099E-0001, 3.91160000000127E-0001, 1.62923000000092E-0001,
  529. -1.24134000000026E-0001, -3.68950000000041E-0001, -4.82100000000173E-0001,
  530. -4.19299000000137E-0001, -2.00237999999899E-0001, 9.56523999999490E-0002,
  531. 3.56715999999778E-0001, 4.80940999999802E-0001, 4.16690999999901E-0001,
  532. 1.85821000000033E-0001, -1.21634999999969E-0001, -3.81190000000061E-0001,
  533. -4.84109999999873E-0001, -3.84040999999797E-0001, -1.19752999999946E-0001,
  534. 1.97697999999946E-0001, 4.30369000000155E-0001, 4.73501999999826E-0001,
  535. 3.04200000000037E-0001, -4.45780999999812E-0003, -3.12843000000157E-0001,
  536. -4.77105999999822E-0001, -4.17065000000093E-0001, -1.57408999999916E-0001,
  537. 1.79722999999967E-0001, 4.30503999999928E-0001, 4.69046000000162E-0001,
  538. 2.72319000000152E-0001, -6.37966000000461E-0002, -3.68903000000046E-0001,
  539. -4.83776999999918E-0001, -3.44759999999951E-0001, -2.16183999999942E-0002,
  540. 3.14663999999993E-0001, 4.81103000000076E-0001, 3.83209000000079E-0001,
  541. 7.12525999999798E-0002, -2.82396000000062E-0001, -4.76846999999907E-0001,
  542. -3.97513999999774E-0001, -8.63047999999935E-0002, 2.77078999999958E-0001,
  543. 4.77041000000099E-0001, 3.90680999999859E-0001, 6.61678999999822E-0002,
  544. -3.00528000000213E-0001, -4.82100000000173E-0001, -3.61706000000140E-0001,
  545. -1.12785000000031E-0002, 3.47796000000017E-0001, 4.82912999999826E-0001,
  546. 3.02329000000100E-0001, -7.90537999999970E-0002, -4.09227999999985E-0001,
  547. -4.64552999999796E-0001, -2.03408999999965E-0001, 1.98789999999917E-0001,
  548. 4.64230000000043E-0001, 4.04536999999891E-0001, 5.75837999999749E-0002,
  549. -3.32002000000102E-0001, -4.82986999999866E-0001, -2.81882000000223E-0001,
  550. 1.27741000000015E-0001, 4.43717999999990E-0001, 4.27357000000029E-0001,
  551. 8.67075999999543E-0002, -3.21608000000197E-0001, -4.82884000000013E-0001,
  552. -2.68137000000024E-0001, 1.58376000000089E-0001, 4.60480000000189E-0001,
  553. 3.94394999999804E-0001, 8.97358999999653E-0003, -3.85126000000128E-0001,
  554. -4.63401999999860E-0001, -1.56926999999996E-0001, 2.82047000000148E-0001,
  555. 4.83843999999863E-0001, 2.73432999999841E-0001, -1.73019000000068E-0001,
  556. -4.70732000000226E-0001, -3.57265000000098E-0001, 7.19904999999699E-0002,
  557. 4.38375000000178E-0001, 4.12060999999994E-0001, 1.23822000000047E-0002,
  558. -3.99836000000050E-0001, -4.45055000000139E-0001, -7.71581999999853E-0002,
  559. 3.63976000000093E-0001, 4.62701999999808E-0001, 1.21280999999954E-0001,
  560. -3.37500999999975E-0001, -4.71152000000075E-0001, -1.45841000000019E-0001,
  561. 3.23628000000099E-0001, 4.73699000000124E-0001, 1.51135999999951E-0001,
  562. -3.24419000000034E-0001, -4.72220000000107E-0001, -1.37774000000036E-0001,
  563. 3.39390999999978E-0001, 4.65411000000131E-0001, 1.04878999999983E-0001,
  564. -3.67131999999856E-0001, -4.50452000000041E-0001, -5.20331000000169E-0002,
  565. 4.03378999999859E-0001, 4.21691999999894E-0001, -2.14152000000070E-0002,
  566. -4.41976000000068E-0001, -3.72757999999976E-0001, 1.13511000000017E-0001,
  567. 4.72905000000083E-0001, 2.96350999999959E-0001, -2.19368000000031E-0001,
  568. -4.83674999999948E-0001, -1.87777999999980E-0001, 3.27948999999990E-0001,
  569. 4.59028000000217E-0001, 4.67350000000124E-0002, -4.22157999999854E-0001,
  570. -3.85088999999880E-0001, 1.17635999999948E-0001, 4.78071999999884E-0001,
  571. 2.53111999999874E-0001, -2.84720999999990E-0001, -4.69815999999810E-0001,
  572. -6.75424999999450E-0002, 4.20501000000058E-0001, 3.76236999999946E-0001,
  573. -1.48369000000002E-0001, -4.83322999999928E-0001, -1.93813000000091E-0001,
  574. 3.48465000000033E-0001, 4.35011999999915E-0001, -5.20594999999844E-0002,
  575. -4.70860999999786E-0001, -2.61331000000155E-0001, 2.99838999999793E-0001,
  576. 4.56122000000050E-0001, -8.47457000000418E-0003, -4.62175000000116E-0001,
  577. -2.79306999999790E-0001, 2.91294999999991E-0001, 4.55973999999969E-0001,
  578. -1.96689999999933E-0002, -4.68189000000166E-0001, -2.51569000000018E-0001,
  579. 3.25217999999950E-0001, 4.34597999999824E-0001, -8.52701999999681E-0002,
  580. -4.81628999999884E-0001, -1.72663999999941E-0001, 3.91235999999935E-0001,
  581. 3.75564999999824E-0001, -2.00489999999945E-0001, -4.76438000000144E-0001,
  582. -3.35570999999959E-0002, 4.60794000000078E-0001, 2.51975000000130E-0001,
  583. -3.44184000000041E-0001, -4.09337999999934E-0001, 1.61409999999933E-0001,
  584. 4.80509999999867E-0001, 4.47993000000224E-0002, -4.62160999999924E-0001,
  585. -2.35175999999910E-0001, 3.67765999999847E-0001, 3.80204000000049E-0001,
  586. -2.21663999999919E-0001, -4.63874000000033E-0001, 5.13983000000167E-0002,
  587. 4.82525999999780E-0001, 1.17171999999982E-0001, -4.42914000000201E-0001,
  588. -2.64184000000114E-0001, 3.57844999999998E-0001, 3.76622000000225E-0001,
  589. -2.43244999999888E-0001, -4.48910999999953E-0001, 1.14519999999970E-0001,
  590. 4.81073999999808E-0001, 1.47939000000008E-0002, -4.77703000000020E-0001,
  591. -1.34743000000071E-0001, 4.45565999999872E-0001, 2.38581000000067E-0001,
  592. -3.92727999999806E-0001, -3.23120999999901E-0001, 3.26771999999892E-0001,
  593. 3.87517000000116E-0001, -2.54672999999912E-0001, -4.33148000000074E-0001,
  594. 1.81812000000036E-0001, 4.62328000000070E-0001, -1.12458999999944E-0001,
  595. -4.78289000000132E-0001, 4.92869999999925E-0002, 4.84148000000005E-0001,
  596. 5.86423000000025E-0003, -4.83146999999917E-0001, -5.22940999999832E-0002,
  597. 4.77914999999939E-0001, 8.96437999999762E-0002, -4.70905000000130E-0001,
  598. -1.18200000000002E-0001, 4.63843999999881E-0001, 1.38171000000057E-0001,
  599. -4.58263999999872E-0001, -1.50045999999975E-0001, 4.55015999999887E-0001,
  600. 1.53954999999996E-0001, -4.54807999999957E-0001, -1.50088000000096E-0001,
  601. 4.57700999999815E-0001, 1.38134999999920E-0001, -4.63569999999891E-0001,
  602. -1.17764999999963E-0001, 4.71551999999974E-0001, 8.81894999999986E-0002,
  603. -4.80403000000024E-0001, -4.87550000000283E-0002, 4.87923000000137E-0001,
  604. -1.28841999999985E-0003, -4.91308000000117E-0001, 6.19993000000250E-0002,
  605. 4.86694000000170E-0001, -1.32732999999917E-0001, -4.69751999999971E-0001,
  606. 2.11195000000089E-0001, 4.35657999999876E-0001, -2.93380999999954E-0001,
  607. -3.80208000000039E-0001, 3.72847999999976E-0001, 3.00403000000188E-0001,
  608. -4.41198999999870E-0001, -1.96052999999893E-0001, 4.88281999999799E-0001,
  609. 7.05491999999595E-0002, -5.03888000000188E-0001, 6.79192000000057E-0002,
  610. 4.79287999999997E-0001, -2.06691999999975E-0001, -4.09948999999870E-0001,
  611. 3.29295000000002E-0001, 2.97392999999829E-0001, -4.18125000000146E-0001,
  612. -1.51035000000093E-0001, 4.57456999999977E-0001, -1.21829999999932E-0002,
  613. -4.37680000000000E-0001, 1.69366000000082E-0001, 3.58297000000221E-0001,
  614. -2.95907000000170E-0001, -2.30043000000023E-0001, 3.70323999999982E-0001,
  615. 7.37844000000223E-0002, -3.79878000000190E-0001, 8.26819999999771E-0002,
  616. 3.24047000000064E-0001, -2.10759999999937E-0001, -2.15877000000091E-0001,
  617. 2.87343999999848E-0001, 7.92186999999558E-0002, -3.00759000000198E-0001,
  618. 5.63948000000210E-0002, 2.53298000000086E-0001, -1.63127999999915E-0001,
  619. -1.60753000000113E-0001, 2.21193999999969E-0001, 4.78082000000200E-0002,
  620. -2.23539999999957E-0001, 5.81391000000053E-0002, 1.76492000000053E-0001,
  621. -1.34119999999939E-0001, -9.74677000000383E-0002, 1.66602999999895E-0001,
  622. 9.46428999999682E-0003, -1.54275999999982E-0001, 6.49469999999610E-0002,
  623. 1.06955999999968E-0001, -1.09595000000013E-0001, -4.21787000000222E-0002,
  624. 1.18017000000009E-0001, -2.05329000000063E-0002, -9.43591999999853E-0002,
  625. 6.49757999999565E-0002, 5.07000999999718E-0002, -8.26296000000184E-0002,
  626. -2.79875000000018E-0003, 7.37881000000016E-0002, -3.52417999999943E-0002,
  627. -4.63371999999822E-0002, 5.46148000000244E-0002, 1.19423999999952E-0002,
  628. -5.37128000000280E-0002, 1.78473999999937E-0002, 3.70907000000216E-0002,
  629. -3.52100999999720E-0002, -1.32136000000003E-0002, 3.75063999999838E-0002,
  630. -8.99504000000206E-0003, -2.73909000000003E-0002, 2.27873000000045E-0002,
  631. 1.08742000000035E-0002, -2.56093000000135E-0002, 5.03203000000241E-0003,
  632. 1.90342999999871E-0002, -1.51086999999990E-0002, -7.63514999999870E-0003,
  633. 1.72402000000034E-0002, -3.46168000000091E-0003, -1.26917999999989E-0002,
  634. 1.03507000000036E-0002, 4.70225000000113E-0003, -1.16266999999937E-0002,
  635. 2.92620999999826E-0003, 8.12958999999580E-0003, -7.45025999999882E-0003,
  636. -2.37827999999851E-0003, 7.79329000000217E-0003, -2.83968999999828E-0003,
  637. -4.85918999999768E-0003, 5.45333000000170E-0003, 6.50574999999876E-0004,
  638. -5.05228000000102E-0003, 2.66608000000090E-0003, 2.55376999999868E-0003,
  639. -3.93425000000036E-0003, 3.76394999999974E-0004, 3.09312999999989E-0003,
  640. -2.39051000000146E-0003, -1.06363999999992E-0003, 2.75651000000110E-0003,
  641. -9.89122000000009E-0004, -1.69909999999973E-0003, 2.01841000000158E-0003,
  642. 2.12075999999894E-0005, -1.75568000000048E-0003, 1.21378999999955E-0003,
  643. 6.16308999999760E-0004, -1.49133999999940E-0003, 5.30142999999761E-0004,
  644. 8.80323999999710E-0004, -1.11185999999996E-0003, 0.00000000000000E+0000
  645. };
  646. Double g_pdGCRSignal1_47[GCR_SIZE] = { 0.0, //TODO - Fix this if we re-enable GCR usage
  647. };
  648. Double g_pdGCRSignal1_4[GCR_SIZE] = {
  649. -0.00347677, -0.00322337, -0.00325605,
  650. -0.00322112, -0.00321719, -0.00320921,
  651. -0.00318706, -0.0031813, -0.00316357,
  652. -0.0031458, -0.0031343, -0.00311177,
  653. -0.00309477, -0.00307598, -0.00305072,
  654. -0.00303091, -0.00300454, -0.00297671,
  655. -0.00295068, -0.00291709, -0.00288493,
  656. -0.00284988, -0.00280893, -0.00276923,
  657. -0.00272296, -0.00267316, -0.00262177,
  658. -0.00256212, -0.00250011, -0.00243231,
  659. -0.00235619, -0.00227647, -0.00218688,
  660. -0.00208914, -0.00198407, -0.00186586,
  661. -0.00173831, -0.00159776, -0.00144136,
  662. -0.00127163, -0.00108236, -0.000873557,
  663. -0.000644226, -0.000387946, -0.000105821,
  664. 0.00020736, 0.000556413, 0.000941966,
  665. 0.00137233, 0.00185082, 0.00238212,
  666. 0.00297575, 0.00363537, 0.00437036,
  667. 0.00519034, 0.00610175, 0.00711803,
  668. 0.00824878, 0.00950516, 0.0109028,
  669. 0.0124523, 0.0141696, 0.0160703,
  670. 0.0181666, 0.0204768, 0.0230134,
  671. 0.0257895, 0.0288187, 0.0321075,
  672. 0.0356624, 0.0394846, 0.0435661,
  673. 0.047895, 0.0524468, 0.0571852,
  674. 0.0620638, 0.0670147, 0.0719563,
  675. 0.0767863, 0.0813785, 0.0855895,
  676. 0.0892493, 0.0921681, 0.0941404,
  677. 0.0949423, 0.0943457, 0.0921242,
  678. 0.0880623, 0.0819759, 0.0737222,
  679. 0.0632224, 0.050484, 0.0356143,
  680. 0.018848, 0.000557361, -0.0187374,
  681. -0.0383613, -0.0575002, -0.07522,
  682. -0.0905034, -0.102312, -0.109649,
  683. -0.111652, -0.107688, -0.0974488,
  684. -0.0810454, -0.0590807, -0.0326872,
  685. -0.00352681, 0.026276, 0.0542588,
  686. 0.0778138, 0.0944444, 0.102052,
  687. 0.0992456, 0.0856211, 0.0619632,
  688. 0.0303378, -0.0059827, -0.0427809,
  689. -0.0753343, -0.0990286, -0.110051,
  690. -0.106085, -0.0868866, -0.0545886,
  691. -0.013663, 0.0295587, 0.0677864,
  692. 0.0939929, 0.102784, 0.0916517,
  693. 0.0618112, 0.0183872, -0.030241,
  694. -0.0739181, -0.102805, -0.109703,
  695. -0.0920499, -0.0530257, -0.00135532,
  696. 0.0503947, 0.0887573, 0.102941,
  697. 0.088103, 0.0472938, -0.00868865,
  698. -0.0638268, -0.101342, -0.108907,
  699. -0.0830246, -0.0309566, 0.0308085,
  700. 0.081466, 0.102895, 0.0864348,
  701. 0.0368566, -0.0283508, -0.0847231,
  702. -0.10992, -0.0928818, -0.0392512,
  703. 0.0297409, 0.0852003, 0.102632,
  704. 0.0731864, 0.00880831, -0.0618224,
  705. -0.105668, -0.100953, -0.0486301,
  706. 0.0265581, 0.0870846, 0.101275,
  707. 0.060433, -0.0149953, -0.0847433,
  708. -0.109925, -0.0751765, 0.00101709,
  709. 0.0750949, 0.102915, 0.0665191,
  710. -0.0132255, -0.0874393, -0.108785,
  711. -0.062238, 0.0231743, 0.091318,
  712. 0.0955987, 0.0316556, -0.0572763,
  713. -0.108566, -0.0844189, -0.000913411,
  714. 0.0811435, 0.0998372, 0.0395636,
  715. -0.0544801, -0.108895, -0.0794151,
  716. 0.0117194, 0.0908163, 0.0917383,
  717. 0.0122175, -0.0811607, -0.107599,
  718. -0.0426033, 0.0575538, 0.103033,
  719. 0.0512332, -0.051541, -0.109666,
  720. -0.0670796, 0.0369271, 0.101949,
  721. 0.0631155, -0.0424742, -0.109048,
  722. -0.0677645, 0.0403094, 0.102837,
  723. 0.0527074, -0.057985, -0.109862,
  724. -0.0449822, 0.0662633, 0.0994283,
  725. 0.0156217, -0.0904555, -0.0957237,
  726. 0.00746482, 0.0979505, 0.0667789,
  727. -0.0498718, -0.109913, -0.0380423,
  728. 0.0773721, 0.090618, -0.0166247,
  729. -0.107162, -0.0624203, 0.0602526,
  730. 0.0982112, -0.000749544, -0.103676,
  731. -0.0691827, 0.0565363, 0.09844,
  732. -0.00374564, -0.105781, -0.0605572,
  733. 0.0678068, 0.0916978, -0.0254848,
  734. -0.109974, -0.0338235, 0.0886854,
  735. 0.0696882, -0.0628569, -0.102695,
  736. 0.0140484, 0.103037, 0.0210051,
  737. -0.100828, -0.0637215, 0.072721,
  738. 0.0826481, -0.0517958, -0.104731,
  739. 0.014821, 0.103036, 0.00681728,
  740. -0.107478, -0.0388149, 0.0925919,
  741. 0.0522746, -0.0885787, -0.0749478,
  742. 0.0695205, 0.0793948, -0.0647939,
  743. -0.0942172, 0.047579, 0.0923407,
  744. -0.0464317, -0.102366, 0.033887,
  745. 0.0969906, -0.0380317, -0.104437,
  746. 0.031087, 0.0969579, -0.040917,
  747. -0.102302, 0.0395639, 0.0922497,
  748. -0.0546754, -0.0940965, 0.0579833,
  749. 0.0792467, -0.0766651, -0.074754,
  750. 0.0818012, 0.0520243, -0.0997399,
  751. -0.0384754, 0.100644, 0.00639079,
  752. -0.110028, 0.0153267, 0.0976144,
  753. -0.0523014, -0.0892546, 0.0731407,
  754. 0.0561963, -0.101035, -0.0273019,
  755. 0.10298, -0.0216741, -0.102356,
  756. 0.0564077, 0.0691551, -0.0959579,
  757. -0.033156, 0.103057, -0.0261942,
  758. -0.0984214, 0.0684994, 0.0529683,
  759. -0.106231, -0.00223094, 0.0983575,
  760. -0.0646024, -0.0682985, 0.097313,
  761. -0.00218509, -0.104729, 0.0614701,
  762. 0.0539863, -0.1072, 0.0111803,
  763. 0.0893852, -0.084751, -0.0365265,
  764. 0.102008, -0.0503339, -0.0726031,
  765. 0.0973383, -0.0146527, -0.0952283,
  766. 0.0831772, 0.0159482, -0.106912,
  767. 0.0664624, 0.0389454, -0.111695,
  768. 0.0521044, 0.0540878, -0.113279,
  769. 0.043042, 0.0618967, -0.114071,
  770. 0.0406873, 0.0626912, -0.114795,
  771. 0.0452189, 0.0562661, -0.114465,
  772. 0.0555428, 0.0422166, -0.110703,
  773. 0.069055, 0.0207763, -0.100524,
  774. 0.0816043, -0.00616025, -0.0816649,
  775. 0.0881263, -0.0344248, -0.0541472,
  776. 0.0841699, -0.0580354, -0.0212963,
  777. 0.0678992, -0.0709826, 0.0106367,
  778. 0.0415202, -0.0697234, 0.0345279,
  779. 0.0110249, -0.0550094, 0.0452444,
  780. -0.0159388, -0.0318536, 0.0417614,
  781. -0.0330188, -0.00759705, 0.0274425,
  782. -0.0374128, 0.0108287, 0.0085779,
  783. -0.0305831, 0.0192502, -0.00799957,
  784. -0.0174139, 0.0175002, -0.0174261,
  785. -0.00409323, 0.00907835, -0.0184854,
  786. 0.0045466, -0.000846416, -0.0134708,
  787. 0.00677922, -0.00796784, -0.00638725,
  788. 0.00390153, -0.0103627, -0.0008869,
  789. -0.0010817, -0.00865121, 0.00113132,
  790. -0.00511521, -0.00515852, 1.14208e-05,
  791. -0.00659004, -0.00231654, -0.00244116,
  792. -0.00577361, -0.00133297, -0.00441284,
  793. -0.00400062, -0.00202788, -0.00497207,
  794. -0.00267527, -0.0033336, -0.00432329,
  795. -0.00246436, -0.00415329, -0.00339386,
  796. -0.00304039, -0.00414497, -0.00292314,
  797. -0.00369477, -0.00363559, -0.00308028,
  798. -0.00390425, -0.00320941, -0.00350326,
  799. -0.00366609, -0.00319138, -0.0037218,
  800. -0.00347683
  801. };
  802. Double g_pdGCRSignal1[MAX_NABTS_SAMPLES_PER_LINE]; /* used for resampling */
  803. int g_nNabtsGcrSize = GCR_SIZE;
  804. /* This array is filled with the inverted GCR waveform. (in FillGCRSignals()) */
  805. Double g_pdGCRSignal2[MAX_NABTS_SAMPLES_PER_LINE];
  806. /* The EqualizeMatch structures for positive GCR, negative GCR, and
  807. NABTS sync.
  808. Each structure contains the appropriate waveform for matching, its size,
  809. its sample rate (actually, individual sample period in units of the
  810. 5*NABTS bit rate), and the minimum and maximum samples among which the
  811. signal may be matched in the VBI samples.
  812. */
  813. EqualizeMatch eqmatchGCR1 = {
  814. GCR_SIZE,
  815. GCR_SAMPLE_RATE,
  816. GCR_START_DETECT,
  817. GCR_END_DETECT,
  818. g_pdGCRSignal1
  819. };
  820. EqualizeMatch eqmatchGCR2 = {
  821. GCR_SIZE,
  822. GCR_SAMPLE_RATE,
  823. GCR_START_DETECT,
  824. GCR_END_DETECT,
  825. g_pdGCRSignal2
  826. };
  827. EqualizeMatch eqmatchNabtsSync = {
  828. NABSYNC_SIZE, /* changes */
  829. NABSYNC_SAMPLE_RATE, /* doesn't change */
  830. NABSYNC_START_DETECT, /* changes */
  831. NABSYNC_END_DETECT, /* changes */
  832. g_pdSync5 /* changes */
  833. };
  834. int pnGCRPositiveIndices0[]= {
  835. 192, 194, 196, 198, 200,
  836. 202, 204, 206, 208, 210,
  837. 212, 214, 216, 218, 220,
  838. 222, 278, 280, 282, 284,
  839. 286, 288, 322, 324, 326,
  840. 328
  841. };
  842. /* Some sample locations at which a "positive" GCR signal should be
  843. quite negative */
  844. int pnGCRNegativeIndices0[]= {
  845. 246, 248, 250, 252, 254,
  846. 256, 258, 260, 262, 302,
  847. 304, 306, 308, 310, 338,
  848. 340, 342, 344, 346
  849. };
  850. int pnGCRPositiveIndices[26];
  851. int pnGCRNegativeIndices[19];
  852. /*******************/
  853. /*** End Globals ***/
  854. /*******************/
  855. /*************************************/
  856. /*** Begin inline helper functions ***/
  857. /*************************************/
  858. /* Inline helper functions for computing min, max, absolute value */
  859. inline Double dmin(Double x, Double y) { return x<y?x:y; }
  860. inline Double dmin3(Double x, Double y, Double z) { return dmin(x,dmin(y,z)); }
  861. inline int imin(int x, int y) { return x<y?x:y; }
  862. inline int imin3(int x, int y, int z) { return imin(x,imin(y,z)); }
  863. inline Double dmax(Double x, Double y) { return x>y?x:y; }
  864. inline Double dmax3(Double x, Double y, Double z) { return dmax(x,dmax(y,z)); }
  865. inline int imax(int x, int y) { return x>y?x:y; }
  866. inline int imax3(int x, int y, int z) { return imax(x,imax(y,z)); }
  867. inline int iabs(int x) { return x>0?x:-x; }
  868. /* idivceil returns a/b, rouding towards positive infinity */
  869. /* idivfloor returns a/b, rounding towards negative infinity */
  870. /* idivceil and idivfloor are somewhat disgusting because
  871. C does not specify rounding behavior with division using negative
  872. numbers (!). */
  873. inline int idivceil(int a, int b)
  874. {
  875. int sgn= 0;
  876. if (a<0) { a= (-a); sgn ^= 1; }
  877. if (b<0) { b= (-b); sgn ^= 1; }
  878. if (!sgn) {
  879. /* Answer will be positive so round away from zero */
  880. return (a+b-1)/b;
  881. } else {
  882. /* Answer will be negative so round towards zero */
  883. return -(a/b);
  884. }
  885. }
  886. inline int idivfloor(int a, int b)
  887. {
  888. int sgn= 0;
  889. if (a<0) { a= (-a); sgn ^= 1; }
  890. if (b<0) { b= (-b); sgn ^= 1; }
  891. if (!sgn) {
  892. /* Answer will be positive so round towards zero */
  893. return a/b;
  894. } else {
  895. /* Answer will be negative so round away from zero */
  896. return -((a+b-1)/b);
  897. }
  898. }
  899. /***********************************/
  900. /*** End inline helper functions ***/
  901. /***********************************/
  902. /* Our own ASSERT macro */
  903. #ifdef DEBUG
  904. // Raises X to the Yth power
  905. static long power(long x, long y)
  906. {
  907. long rval;
  908. long i;
  909. rval = x;
  910. for (i = 2; i <= y; ++i)
  911. rval *= x;
  912. return (rval);
  913. }
  914. static char *flPrintf(Double dNum, int nPrec)
  915. {
  916. static char rbufs[4][50];
  917. static short rbuf = 0;
  918. long lFix, lNum, lFrac;
  919. char *rval = rbufs[rbuf++];
  920. char *ep;
  921. lFix = float2long(dNum * power(10, nPrec));
  922. lNum = lFix / power(10, nPrec);
  923. lFrac = lFix % power(10, nPrec);
  924. if (lFrac < 0)
  925. lFrac = -lFrac;
  926. sprintf(rval, "%ld.%0*ld", lNum, nPrec, lFrac);
  927. for (ep = rval; *ep; ++ep) ;
  928. for ( ; ep > rval && ep[-1] != '.'; --ep) {
  929. if (*ep == '0')
  930. *ep = '\0';
  931. }
  932. return (rval);
  933. }
  934. #endif //DEBUG
  935. /* Fill the negative GCR signal, and normalize the various
  936. EqualizeMatch signals */
  937. void NDSPInitGlobals()
  938. {
  939. if (g_bNDSPGlobalsInitted) return;
  940. memcpy(g_pdGCRSignal1, g_pdGCRSignal1_5, GCR_SIZE*sizeof(Double));
  941. FillGCRSignals();
  942. memcpy(g_pdSync, g_pdSync5, NABSYNC_SIZE*sizeof(Double));
  943. NormalizeSyncSignal();
  944. memcpy(pnGCRPositiveIndices, pnGCRPositiveIndices0, 26*sizeof(int));
  945. memcpy(pnGCRNegativeIndices, pnGCRNegativeIndices0, 19*sizeof(int));
  946. g_bNDSPGlobalsInitted= 1;
  947. }
  948. int NDSPStateSetSampleRate(NDSPState* pState, unsigned long newRate)
  949. {
  950. double newMultiple;
  951. unsigned long oldRate;
  952. debug_printf(("NDSPStateSetSampleRate(0x%p, %lu) entered\n", pState, newRate));
  953. newMultiple = (double)newRate / KS_VBIDATARATE_NABTS; // (28636360 / 5727272.0) = 5.0
  954. switch (newRate) {
  955. case KS_VBISAMPLINGRATE_4X_NABTS: // 4X NABTS
  956. g_filterDefault = &g_filterDefault4;
  957. break;
  958. case KS_VBISAMPLINGRATE_47X_NABTS: // 4.7X NABTS
  959. g_filterDefault = &g_filterDefault47;
  960. break;
  961. case KS_VBISAMPLINGRATE_5X_NABTS: // 5X NABTS
  962. g_filterDefault = &g_filterDefault5;
  963. break;
  964. /* ===== add cases here as necessary ===== */
  965. default:
  966. debug_printf(("Unsupported sample rate: %luhz\n", newRate));
  967. return NDSP_ERROR_UNSUPPORTED_SAMPLING_RATE;
  968. }
  969. NDSPStateSetFilter(pState, g_filterDefault, sizeof(*g_filterDefault));
  970. pState->dSampleRate = newMultiple;
  971. oldRate = float2long(pState->dSampleFreq);
  972. pState->dSampleFreq = newRate;
  973. switch (newRate) {
  974. case KS_VBISAMPLINGRATE_4X_NABTS:
  975. ClearVariableFIRFilter(&pState->filterGCREqualizeTemplate);
  976. AddToVariableFIRFilter(&pState->filterGCREqualizeTemplate,
  977. 0, 11, 9, 2);
  978. break;
  979. case KS_VBISAMPLINGRATE_47X_NABTS:
  980. ClearVariableFIRFilter(&pState->filterGCREqualizeTemplate);
  981. AddToVariableFIRFilter(&pState->filterGCREqualizeTemplate,
  982. 0, 11, 9, 2); //FIXME
  983. break;
  984. case KS_VBISAMPLINGRATE_5X_NABTS:
  985. break;
  986. /* ===== add cases here as necessary ===== */
  987. default:
  988. // Unknown sample rate
  989. EASSERT(0);
  990. break;
  991. }
  992. /* Last but not least, recompute for the new rate */
  993. NDSPComputeNewSampleRate(newRate, oldRate);
  994. return 0;
  995. }
  996. /* Create a new "DSP state".
  997. A separate DSP state should be maintained for each separate
  998. simultaneous source to the DSP.
  999. NDSPStartRetrain() is implicitly called upon creation of a new state.
  1000. Public function: also see nabtsapi.h.
  1001. If memory is passed, use it for storing the state rather than mallocing
  1002. a new state */
  1003. NDSPState *NDSPStateNew(void *memory)
  1004. {
  1005. NDSPState *pState;
  1006. /* Be sure that the globals are initialized */
  1007. NDSPInitGlobals();
  1008. pState= memory ? ((NDSPState *)memory) : alloc_mem(sizeof(NDSPState));
  1009. if (!pState) return NULL;
  1010. memset(pState, 0, sizeof(NDSPState));
  1011. /* Magic number to help identify an NDSPState object */
  1012. pState->uMagic= NDSP_STATE_MAGIC;
  1013. pState->nRetrainState= 0;
  1014. pState->nUsingGCR= 0;
  1015. pState->bUsingScratch1= FALSE;
  1016. pState->bUsingScratch2= FALSE;
  1017. pState->bUsingScratch3= FALSE;
  1018. pState->bUsingScratch4= FALSE;
  1019. pState->bUsingScratch5= FALSE;
  1020. pState->bFreeStateMem = (memory == NULL);
  1021. pState->SamplesPerLine= 1600; // Default from Bt829 driver
  1022. /* Reset the filter to be the "fallback" filter */
  1023. NDSPResetFilter(pState);
  1024. /* Set the equalization "template" filters.
  1025. (These filters have the correct tap locations, but the tap
  1026. values are ignored) */
  1027. ClearVariableFIRFilter(&pState->filterGCREqualizeTemplate);
  1028. AddToVariableFIRFilter(&pState->filterGCREqualizeTemplate,
  1029. 0, 11, 9, 3);
  1030. ClearVariableFIRFilter(&pState->filterNabsyncEqualizeTemplate);
  1031. AddToVariableFIRFilter(&pState->filterNabsyncEqualizeTemplate,
  1032. 0, 2, 2, 5);
  1033. NDSPStateSetSampleRate(pState, KS_VBISAMPLINGRATE_5X_NABTS);
  1034. return pState;
  1035. }
  1036. /* Destroys the DSP state.
  1037. Automatically disconnects from any FEC state.
  1038. Returns 0 on success.
  1039. Returns NDSP_ERROR_ILLEGAL_NDSP_STATE if illegal state.
  1040. Public function: also see nabtsapi.h */
  1041. int NDSPStateDestroy(NDSPState *pState)
  1042. {
  1043. if (!pState || pState->uMagic != NDSP_STATE_MAGIC)
  1044. return NDSP_ERROR_ILLEGAL_NDSP_STATE;
  1045. if (pState->bFreeStateMem)
  1046. free_mem(pState);
  1047. return 0;
  1048. }
  1049. /*
  1050. Connect the given NFECState and NDSPState
  1051. For cases where the NDSP and NFEC modules are connected,
  1052. giving pointers to the connected state may result in increased
  1053. robustness and efficiency.
  1054. Note that only one of
  1055. NDSPStateConnectToFEC or
  1056. NFECStateConnectToDSP
  1057. need be called to connect the two states. (Calling both is OK).
  1058. Returns 0 on success.
  1059. Returns NDSP_ERROR_ILLEGAL_NDSP_STATE if illegal DSP state.
  1060. Public function: also see nabtsapi.h
  1061. This function currently does nothing, but the API is in place
  1062. for future algorithms that could potentially make use of combined
  1063. DSP and FEC knowledge */
  1064. int NDSPStateConnectToFEC(NDSPState *pDSPState, NFECState *pFECState)
  1065. {
  1066. if (!pDSPState || pDSPState->uMagic != NDSP_STATE_MAGIC)
  1067. return NDSP_ERROR_ILLEGAL_NDSP_STATE;
  1068. return 0;
  1069. }
  1070. /*
  1071. Tells the DSP to initiate a "fast retrain". This is useful if you
  1072. suspect that conditions have changed sufficiently to be worth spending
  1073. significant CPU to quickly train on a signal.
  1074. This should be called when the source of video changes.
  1075. Returns 0 on success.
  1076. Returns NDSP_ERROR_ILLEGAL_NDSP_STATE if illegal DSP state.
  1077. Public function: also see nabtsapi.h
  1078. */
  1079. int NDSPStartRetrain(NDSPState *pState)
  1080. {
  1081. if (!pState || pState->uMagic != NDSP_STATE_MAGIC)
  1082. return NDSP_ERROR_ILLEGAL_NDSP_STATE;
  1083. /* Reset nRetrainState to zero.
  1084. This initiates a retrain.
  1085. See NDSPProcessGCRLine for more information */
  1086. pState->nRetrainState= 0;
  1087. return 0;
  1088. }
  1089. /*
  1090. Resets the DSP filter back to the "fallback" fixed filter.
  1091. This normally doesn't need to be called, but might be useful for
  1092. performance analysis.
  1093. */
  1094. int NDSPResetFilter(NDSPState *pState)
  1095. {
  1096. if (!pState || pState->uMagic != NDSP_STATE_MAGIC)
  1097. return NDSP_ERROR_ILLEGAL_NDSP_STATE;
  1098. NDSPStateSetFilter(pState, g_filterDefault, sizeof(*g_filterDefault));
  1099. pState->nRetrainState= 0;
  1100. return 0;
  1101. }
  1102. /* Assign an array of Double from an array of unsigned char */
  1103. void Copy_d_8(Double *pfOutput, unsigned char *pbInput, int nLen)
  1104. {
  1105. int i;
  1106. for (i= 0; i< nLen; i++) pfOutput[i]= pbInput[i];
  1107. }
  1108. /*
  1109. The general strategy for convolution is to "slide" one array across
  1110. another and multiple the overlapping portions like so:
  1111. Convolve A and B by sliding B across A.
  1112. -- j
  1113. \
  1114. dest[i]= / a[j*a_step+i]*b[j*b_step]
  1115. --
  1116. For Convolve_d_d_filt_var, we are convolving and array of Double
  1117. with a variable-tap filter. A variable-tap filter is simply a sparse
  1118. representation for an array */
  1119. int debug_convolve= 0;
  1120. void Convolve_d_d_filt_var(NDSPState *pState,
  1121. Double *dest, int dest_begin, int dest_end,
  1122. Double *a, int a_begin, int a_end,
  1123. FIRFilter *pFilt,
  1124. BOOL bSubtractMean) {
  1125. int maxTap = pFilt->nMaxTap; /* positive value */
  1126. int minTap = pFilt->nMinTap; /* negative value */
  1127. int b_len = maxTap-minTap+1;
  1128. Double *b;
  1129. int i;
  1130. /* Set b to a scratch buffer from the NDSPState.
  1131. Mark the scratch buffer as used.
  1132. Offset b by "minTap" (which is negative), so that array references
  1133. as low as "minTap" can be handled.
  1134. */
  1135. SASSERT(!pState->bUsingScratch7);
  1136. SASSERT(sizeof(pState->pdScratch7)/sizeof(Double) >= b_len);
  1137. b = pState->pdScratch7 - minTap;
  1138. pState->bUsingScratch7 = __LINE__;
  1139. /* The strategy here is to generate a non-sparse representation for the
  1140. filter, and call the standard convolution routine with it. */
  1141. for (i = minTap; i <= maxTap; i++) {
  1142. b[i] = 0;
  1143. }
  1144. for (i = 0; i < pFilt->nTaps; i++) {
  1145. b[pFilt->pnTapLocs[i]] = pFilt->pdTaps[i];
  1146. }
  1147. Convolve_d_d_d(dest, dest_begin, dest_end,
  1148. a, a_begin, a_end, 1,
  1149. b, minTap, maxTap, 1,
  1150. bSubtractMean);
  1151. /* Mark Scratch7 as free */
  1152. pState->bUsingScratch7 = 0;
  1153. }
  1154. /* Like Convolve_d_d_filt_var above, but is implemented to work
  1155. only for fixed-tap filters. This improves performance. */
  1156. void Convolve_d_d_filt(Double *dest, int dest_begin, int dest_end,
  1157. Double *a, int a_begin, int a_end,
  1158. FIRFilter *pFilt,
  1159. BOOL bSubtractMean)
  1160. {
  1161. Double *b= pFilt->pdTaps+(pFilt->nTaps-1)/2;
  1162. Double b_begin= -(pFilt->nTaps-1)/2;
  1163. Double b_end= (pFilt->nTaps-1)/2;
  1164. Double a_step= pFilt->dTapSpacing;
  1165. Double b_step= 1.0;
  1166. SASSERT(pFilt->nTaps > 0 && (pFilt->nTaps % 2) == 1);
  1167. /* Simply call the standard convolution routine, taking into account
  1168. the fixed spacing of the filter */
  1169. Convolve_d_d_d(dest, dest_begin, dest_end,
  1170. a, a_begin, a_end, a_step,
  1171. b, b_begin, b_end, b_step,
  1172. bSubtractMean);
  1173. }
  1174. /*
  1175. Convolve_d_d_d convolves two input arrays and places the result in
  1176. a single output array.
  1177. The general strategy for convolution is to "slide" one array across
  1178. another and multiple the overlapping portions like so:
  1179. Convolve A and B by sliding B across A.
  1180. -- j
  1181. \
  1182. dest[i]= / a[j*a_step+i]*b[j*b_step]
  1183. --
  1184. Convolve_d_d_d complicates matters somewhat by allowing:
  1185. 1) Arbitrary begin and end indices for the destination array.
  1186. 2) Arbitrary begin and end indices for A and B
  1187. (Any elements required by the convolution that fall outside these
  1188. ranges are taken to be zero)
  1189. 3) Arbitrary spacing between the array elements of A and B, allowing
  1190. for a fixed-spacing sparse array to be convolved efficiently.
  1191. 4) The ability to subtract the mean from one of the arguments prior
  1192. to convolution.
  1193. Note that the arbitrary begin and end indices can be negative if
  1194. appropriate.
  1195. */
  1196. void Convolve_d_d_d(Double *dest, int dest_begin, int dest_end,
  1197. Double *a, Double a_begin, Double a_end, Double a_step,
  1198. Double *b, Double b_begin, Double b_end, Double b_step,
  1199. BOOL bSubtractMean)
  1200. {
  1201. Double dSum;
  1202. int i;
  1203. Double j;
  1204. ASSERT( a_step > 0.0 && b_step > 0.0 ); // This should never happen!
  1205. if ( a_step > 0 && b_step > 0 ) // To fix a PREFIX tool divide-by-zero whine
  1206. {
  1207. for (i= dest_begin; i <= dest_end; i++) {
  1208. Double j_begin, j_end;
  1209. Double a_index, b_index;
  1210. Double a_interpolated, b_interpolated;
  1211. /* a_begin <= j*a_step+i <= a_end */
  1212. /* b_begin <= j*b_step <= b_end */
  1213. /* or */
  1214. /* ceil((a_begin-i)/a_step) <= j <= floor((a_end-i)/a_step) */
  1215. /* ceil(b_begin/b_step) <= j <= floor(b_end/b_step) */
  1216. j_begin= max( (a_begin-i) / a_step, (b_begin) / b_step );
  1217. SASSERT((a_begin <= j_begin*a_step+i &&
  1218. b_begin <= j_begin*b_step));
  1219. SASSERT(!(a_begin <= (j_begin-1)*a_step+i &&
  1220. b_begin <= (j_begin-1)*b_step));
  1221. j_end = min( (a_end-i) / a_step, (b_end) / b_step );
  1222. SASSERT((j_end*a_step+i <= a_end &&
  1223. j_end*b_step <= b_end));
  1224. if (debug_convolve) {
  1225. debug_printf(("(%d,%s-%s) ",
  1226. i,
  1227. flPrintf(j_begin,6),
  1228. flPrintf(j_end,6)));
  1229. }
  1230. dSum= 0.0;
  1231. if (j_end >= j_begin) {
  1232. Double dMean;
  1233. a_index= j_begin*a_step+i;
  1234. b_index= j_begin*b_step;
  1235. dMean= 0.0;
  1236. if (bSubtractMean) {
  1237. for(j= j_begin; j <= j_end; j++) {
  1238. SASSERT(b_begin <= b_index && b_index <= b_end);
  1239. b_interpolated = _InterpDoubleArr(b, b_index);
  1240. dMean += b_interpolated;
  1241. b_index += b_step;
  1242. }
  1243. dMean /= (j_end - j_begin + 1);
  1244. }
  1245. a_index= j_begin*a_step+i;
  1246. b_index= j_begin*b_step;
  1247. for(j= j_begin; j <= j_end; j++) {
  1248. SASSERT(a_begin <= a_index && a_index <= a_end);
  1249. SASSERT(b_begin <= b_index && b_index <= b_end);
  1250. a_interpolated = _InterpDoubleArr(a,a_index);
  1251. b_interpolated = _InterpDoubleArr(b,b_index);
  1252. dSum += a_interpolated*(b_interpolated-dMean);
  1253. a_index += a_step;
  1254. b_index += b_step;
  1255. }
  1256. }
  1257. dest[i]= dSum;
  1258. }
  1259. }
  1260. }
  1261. /* Determine the minimum and maximum tap locations for a variable-tap
  1262. filter */
  1263. void RecalcVariableFIRFilterBounds(FIRFilter *pFilter)
  1264. {
  1265. int i;
  1266. int nMinTap= +NDSP_MAX_FIR_TAIL;
  1267. int nMaxTap= -NDSP_MAX_FIR_TAIL;
  1268. SASSERT(pFilter->bVariableTaps);
  1269. for (i= 0; i< pFilter->nTaps; i++) {
  1270. nMinTap= imin(nMinTap, pFilter->pnTapLocs[i]);
  1271. nMaxTap= imax(nMaxTap, pFilter->pnTapLocs[i]);
  1272. }
  1273. if (pFilter->nTaps) {
  1274. pFilter->nMinTap= nMinTap;
  1275. pFilter->nMaxTap= nMaxTap;
  1276. } else {
  1277. pFilter->nMinTap= 0;
  1278. pFilter->nMaxTap= 0;
  1279. }
  1280. }
  1281. /* Adds one or more taps to a variable FIR filter.
  1282. One tap is added at nCenterPos.
  1283. Additional taps are added before or after nCenterPos depending on the
  1284. number of left taps and right taps requested.
  1285. To be more precise, taps are added at all:
  1286. nCenterPos + i * nTapSpacing
  1287. where i is an integer in the range from (-nLeftTaps) to
  1288. nRightTaps, inclusive.
  1289. Will call RecalcVariableFIRFilterBounds to recalculate filter
  1290. bounds in case the tap was added outside previous bounds */
  1291. void AddToVariableFIRFilter(FIRFilter *pFilter,
  1292. int nCenterPos,
  1293. int nLeftTaps, int nRightTaps, int nTapSpacing)
  1294. {
  1295. int i;
  1296. int nCurrentTap= pFilter->nTaps;
  1297. pFilter->bVariableTaps= TRUE;
  1298. /*
  1299. debug_printf(("\nADD %lx [+%d] l=%d r=%d spacing=%d\n",
  1300. (long)pFilter, nCenterPos, nLeftTaps, nRightTaps, nTapSpacing));
  1301. */
  1302. for (i= -nLeftTaps; i <= nRightTaps; i++) {
  1303. int nCurrentTapLoc= nCenterPos + i*nTapSpacing;
  1304. SASSERT(nCurrentTap < NDSP_MAX_FIR_COEFFS);
  1305. SASSERT(iabs(nCurrentTapLoc) <= NDSP_MAX_FIR_TAIL);
  1306. pFilter->pnTapLocs[nCurrentTap]= nCurrentTapLoc;
  1307. pFilter->pdTaps[nCurrentTap]= 0.0;
  1308. nCurrentTap++;
  1309. }
  1310. pFilter->nTaps= nCurrentTap;
  1311. RecalcVariableFIRFilterBounds(pFilter);
  1312. }
  1313. /* Removes all taps from a variable FIR filter */
  1314. void ClearVariableFIRFilter(FIRFilter *pFilter)
  1315. {
  1316. /*debug_printf(("\nCLEAR %lx\n", (long)pFilter));*/
  1317. pFilter->bVariableTaps= TRUE;
  1318. pFilter->nTaps= 0;
  1319. RecalcVariableFIRFilterBounds(pFilter);
  1320. }
  1321. /* Copies a FIRFilter */
  1322. void CopyFIRFilter(FIRFilter *pfilterDest, FIRFilter *pfilterSrc)
  1323. {
  1324. memcpy(pfilterDest, pfilterSrc, sizeof(FIRFilter));
  1325. }
  1326. /* Attempt to perform equalization and long ghost cancellation from
  1327. GCR in the state "pState".
  1328. Assumes that pState->psmPosGCRs[0] has been filled during by
  1329. NDSPProcessGCRLine during retrain states (-g_nNabtsRetrainDuration) to 0.
  1330. (see NDSPProcessGCRLine for more information).
  1331. Return 0 if success. Return error status if failure.
  1332. */
  1333. int DoEqualizeFromGCRs(NDSPState *pState)
  1334. {
  1335. /* If the signal doesn't look enough like a GCR signal, fail
  1336. and say we didn't detect the GCR */
  1337. if (NDSPDetectGCRConfidence(pState->psmPosGCRs[0].pbSamples+
  1338. pState->psmPosGCRs[0].nOffset) < 50) {
  1339. return NDSP_ERROR_NO_GCR;
  1340. }
  1341. return DoGCREqualFromSignal(
  1342. pState,
  1343. &pState->psmPosGCRs[0],
  1344. &eqmatchGCR1,
  1345. &pState->filterGCREqualizeTemplate,
  1346. g_nNabtsAdditionalTapsGCR);
  1347. }
  1348. /* Attempt to perform equalization (WITHOUT long ghost cancellation)
  1349. from GCR in the state "pState".
  1350. Assumes that pState->psmSyncs[0] has been filled during by
  1351. NDSPDecodeLine during retrain states (-g_nNabtsRetrainDuration) to 0.
  1352. (see NDSPProcessGCRLine for more information).
  1353. Return 0 if success. Return error status if failure.
  1354. */
  1355. int DoEqualizeFromNabsync(NDSPState *pState)
  1356. {
  1357. return DoGCREqualFromSignal(
  1358. pState,
  1359. &pState->psmSyncs[0],
  1360. &eqmatchNabtsSync,
  1361. &pState->filterNabsyncEqualizeTemplate,
  1362. 0 /* don't add additional taps for equalizing from nabsync */);
  1363. }
  1364. /* Attempt to perform equalization (WITHOUT long ghost cancellation)
  1365. from given signal and given NDSPSigMatch.
  1366. After equalization, attempt to add nAddTaps taps to handle long ghosts.
  1367. This is only appropriate if the sigmatch signal is long enough.
  1368. Return 0 if success. Return error status if failure.
  1369. */
  1370. int DoGCREqualFromSignal(NDSPState *pState,
  1371. NDSPSigMatch *sigmatch,
  1372. EqualizeMatch *eqm,
  1373. FIRFilter *pfilterTemplate,
  1374. int nAddTaps)
  1375. {
  1376. int i;
  1377. Double dConv;
  1378. Double *GCRsignal= eqm->pdSignal;
  1379. int nMatchSize= eqm->nSignalSize;
  1380. int nMatchSampleRate= eqm->nSignalSampleRate;
  1381. Double *pdSamples;
  1382. int nMaxindex= 0;
  1383. int nStatus= 0;
  1384. if (!pState || pState->uMagic != NDSP_STATE_MAGIC)
  1385. return NDSP_ERROR_ILLEGAL_NDSP_STATE;
  1386. #ifdef DEBUG
  1387. debug_printf(("GCRs: "));
  1388. for (i= 0; i< NDSP_SIGS_TO_ACQUIRE; i++) {
  1389. debug_printf(("%s@%d -%s@%d ",
  1390. flPrintf(pState->psmPosGCRs[i].dMaxval,4),
  1391. pState->psmPosGCRs[i].nOffset,
  1392. flPrintf(pState->psmNegGCRs[i].dMaxval,4),
  1393. pState->psmNegGCRs[i].nOffset));
  1394. }
  1395. debug_printf(("\n"));
  1396. # if 0
  1397. {
  1398. FILE *out= fopen("foo+", "w");
  1399. NDSPAvgGCRs(buf, pState->psmPosGCRs);
  1400. for (i= 0; i< pState->SamplesPerLine; i++) fprintf(out, "%g\n", buf[i]);
  1401. fclose(out);
  1402. }
  1403. {
  1404. FILE *out= fopen("foo-", "w");
  1405. NDSPAvgGCRs(buf, pState->psmNegGCRs);
  1406. for (i= 0; i< pState->SamplesPerLine; i++) fprintf(out, "%g\n", buf[i]);
  1407. fclose(out);
  1408. }
  1409. # endif //0
  1410. #endif //DEBUG
  1411. SASSERT(!pState->bUsingScratch1);
  1412. SASSERT(sizeof(pState->pdScratch1)/sizeof(Double) >= pState->SamplesPerLine);
  1413. pdSamples= pState->pdScratch1;
  1414. pState->bUsingScratch1= __LINE__;
  1415. SASSERT(!pState->bUsingScratch2);
  1416. SASSERT(sizeof(pState->pdScratch2)/sizeof(Double) >= pState->SamplesPerLine);
  1417. if (NDSPAvgSigs(pState, pdSamples, sigmatch, &dConv)) {
  1418. pState->bUsingScratch1= FALSE;
  1419. return NDSP_ERROR_NO_GCR;
  1420. } else {
  1421. debug_printf(("[Conv=%s] ", flPrintf(dConv,4)));
  1422. }
  1423. nMaxindex= 20;
  1424. {
  1425. int nFirstTap;
  1426. int nLastTap;
  1427. Double *pdInput;
  1428. BOOL bRet;
  1429. int nTaps;
  1430. FIRFilter *pfilterGcr= &pState->filterScratch3;
  1431. Double dSyncVal;
  1432. CopyFIRFilter(pfilterGcr, pfilterTemplate);
  1433. SASSERT(pfilterGcr->nTaps > 0);
  1434. nTaps= pfilterGcr->nTaps;
  1435. nFirstTap= pfilterGcr->nMinTap;
  1436. nLastTap= pfilterGcr->nMaxTap;
  1437. SASSERT(!pState->bUsingScratch2);
  1438. pdInput= pState->pdScratch2 - nFirstTap;
  1439. SASSERT(sizeof(pState->pdScratch2)/sizeof(Double) >=
  1440. nMatchSize*nMatchSampleRate-nFirstTap+nLastTap);
  1441. pState->bUsingScratch2= TRUE;
  1442. dSyncVal= Mean_d(pdSamples, 5);
  1443. for (i= nFirstTap;
  1444. i< nMatchSize*nMatchSampleRate+nLastTap;
  1445. i++) {
  1446. if (i+nMaxindex >= 0 && i+nMaxindex < pState->SamplesPerLine) {
  1447. pdInput[i]= pdSamples[i+nMaxindex];
  1448. } else {
  1449. /*debug_printf(("zf %d %s ", i, flPrintf(dSyncVal,4)));*/
  1450. pdInput[i]= dSyncVal;
  1451. }
  1452. }
  1453. SubtractMean_d(pdInput+nFirstTap,
  1454. pdInput+nFirstTap,
  1455. nMatchSize*nMatchSampleRate-nFirstTap+nLastTap);
  1456. /* Perform equalization step */
  1457. bRet= EqualizeVar(pState,
  1458. pdInput, nFirstTap, nMatchSize*nMatchSampleRate+nLastTap,
  1459. GCRsignal, nMatchSize, nMatchSampleRate,
  1460. pfilterGcr);
  1461. {
  1462. int minTap = pfilterGcr->nMinTap;
  1463. int maxTap = pfilterGcr->nMaxTap;
  1464. /* Attempt to add additional taps (if requested) to handle
  1465. long ghosts */
  1466. while (nAddTaps > 0) {
  1467. nAddTaps--;
  1468. if (bRet) {
  1469. Double *pdFilt;
  1470. Double *pdConv;
  1471. int bestTap = -9999;
  1472. SASSERT(!pState->bUsingScratch8);
  1473. SASSERT(sizeof(pState->pdScratch8)/sizeof(Double) >=
  1474. g_nNabtsMaxPreGhost + g_nNabtsMaxPostGhost + 1);
  1475. pdConv = pState->pdScratch8;
  1476. pState->bUsingScratch8 = __LINE__;
  1477. SASSERT(!pState->bUsingScratch6);
  1478. SASSERT(sizeof(pState->pdScratch6)/sizeof(Double) >= pState->SamplesPerLine);
  1479. pdFilt = pState->pdScratch6;
  1480. pState->bUsingScratch6 = __LINE__;
  1481. /* Filter the input line given the best filter so far */
  1482. Convolve_d_d_filt_var(pState,
  1483. pdFilt, 0, pState->SamplesPerLine-1,
  1484. pdInput, 0, pState->SamplesPerLine-1,
  1485. pfilterGcr, TRUE);
  1486. /* Now convolve the filtered output with the target signal
  1487. to see if we can see any echos */
  1488. Convolve_d_d_d(pdConv + g_nNabtsMaxPreGhost,
  1489. -g_nNabtsMaxPreGhost, g_nNabtsMaxPostGhost,
  1490. pdFilt, 0,
  1491. pState->SamplesPerLine-1, GCR_SAMPLE_RATE,
  1492. GCRsignal, 0, g_nNabtsGcrSize, 1,
  1493. TRUE);
  1494. {
  1495. Double bestTapVal = -1;
  1496. /* Look for the strongest echo */
  1497. for (i = 0;
  1498. i < g_nNabtsMaxPreGhost + g_nNabtsMaxPostGhost + 1;
  1499. i++) {
  1500. if (fabs(pdConv[i]) > bestTapVal &&
  1501. (i-g_nNabtsMaxPreGhost < minTap ||
  1502. i-g_nNabtsMaxPreGhost > maxTap)) {
  1503. int j;
  1504. int tap_ok = 1;
  1505. for (j = 0; j < pfilterGcr->nTaps; j++) {
  1506. if (i-g_nNabtsMaxPreGhost == -pfilterGcr->pnTapLocs[j]) {
  1507. tap_ok = 0;
  1508. break;
  1509. }
  1510. }
  1511. if (tap_ok) {
  1512. bestTapVal = fabs(pdConv[i]);
  1513. bestTap = i-g_nNabtsMaxPreGhost;
  1514. }
  1515. }
  1516. }
  1517. }
  1518. debug_printf(("Adding %d to current %d-tap filter\n",
  1519. -bestTap, pfilterGcr->nTaps));
  1520. /* Add a tap to the filter to handle the strongest echo location */
  1521. AddToVariableFIRFilter(pfilterGcr,
  1522. -bestTap,
  1523. 0, 0, 0);
  1524. /* Now re-run the equalization given the new tap locations as
  1525. the prototype */
  1526. bRet= EqualizeVar(pState,
  1527. pdInput, pfilterGcr->nMinTap,
  1528. nMatchSize*nMatchSampleRate+pfilterGcr->nMaxTap,
  1529. GCRsignal, nMatchSize, nMatchSampleRate,
  1530. pfilterGcr);
  1531. pState->bUsingScratch8 = 0;
  1532. pState->bUsingScratch6 = 0;
  1533. }
  1534. }
  1535. }
  1536. if (!bRet) {
  1537. /* Equalization failed */
  1538. nStatus= NDSP_ERROR_NO_GCR;
  1539. NDSPStateSetFilter(pState, g_filterDefault, sizeof(*g_filterDefault));
  1540. pState->nUsingGCR= FALSE;
  1541. } else {
  1542. nStatus= 0;
  1543. NDSPStateSetFilter(pState, pfilterGcr, sizeof(*pfilterGcr));
  1544. pState->nUsingGCR= TRUE;
  1545. }
  1546. pdInput= NULL;
  1547. pState->bUsingScratch2= FALSE;
  1548. }
  1549. pdSamples= NULL;
  1550. pState->bUsingScratch1= FALSE;
  1551. return nStatus;
  1552. }
  1553. /* Do we want to use the actual GCR signal for equalization?
  1554. If FALSE, fall back to only use NABTS sync equalization */
  1555. BOOL g_bUseGCR= FALSE; // Can't use GCR for EQ right now due to IP concerns
  1556. int DoEqualize(NDSPState *pState)
  1557. {
  1558. /* Do equalization based on either GCR acquisition,
  1559. or Nabsync acquisition */
  1560. /* Preference is given to GCR, but only if
  1561. g_bUseGCR is TRUE */
  1562. if (g_bUseGCR) {
  1563. if (0 == DoEqualizeFromGCRs(pState)) {
  1564. #ifdef DEBUG
  1565. debug_printf(("GCR equalize "));
  1566. print_filter(&pState->filter);
  1567. debug_printf(("\n"));
  1568. #endif //DEBUG
  1569. return 0;
  1570. } else {
  1571. debug_printf(("GCR equ failed\n"));
  1572. }
  1573. }
  1574. /* GCR failed or was disabled. Try NABTS sync */
  1575. /* toggle default filtering */
  1576. if (0 == DoEqualizeFromNabsync(pState)) {
  1577. #ifdef DEBUG
  1578. debug_printf(("Nabsync equalize "));
  1579. print_filter(&pState->filter);
  1580. debug_printf(("\n"));
  1581. #endif //DEBUG
  1582. return 0;
  1583. } else {
  1584. debug_printf(("Nabsync equ failed\n"));
  1585. }
  1586. /* All adaptive equalization failed;
  1587. use default fixed filter */
  1588. debug_printf(("Using default filter\n"));
  1589. NDSPStateSetFilter(pState, g_filterDefault, sizeof(*g_filterDefault));
  1590. #ifdef DEBUG
  1591. print_filter(&(pState->filter));
  1592. #endif //DEBUG
  1593. return 1;
  1594. }
  1595. #define NABTS_RATE KS_VBIDATARATE_NABTS
  1596. #define NABTS_SAMPLE_RATE (NABTS_RATE * 5)
  1597. #define NABTS_US_PER_CLOCK_TIMES_100000 (1000000000 / (NABTS_SAMPLE_RATE/100))
  1598. /*
  1599. ComputeOffsets takes a VBIINFOHEADER and determines the correct offset
  1600. into the samples for the DSP code.
  1601. The sample offset is returned in both pnOffsetData and
  1602. pnOffsetSamples. Only one is now required; we should remove one of
  1603. the two arguments at earliest convenience.
  1604. Our goal is to have the nabts sync start around sample 52,
  1605. or 1.82 us into the block.
  1606. NABTS sync is specified to be inserted at 10.48 +- .34 us after
  1607. leading edge of HSYNC.
  1608. Therefore, our desired sample start time is 10.48 - 1.82 = 8.66 us
  1609. after leading edge of HSYNC
  1610. */
  1611. #define DESIRED_START_TIME_US100 866
  1612. #if 0
  1613. int ComputeOffsets(KS_VBIINFOHEADER *pVBIINFO,
  1614. int *pnOffsetData,
  1615. int *pnOffsetSamples)
  1616. {
  1617. int nOffsetUS100;
  1618. int nOffsetData;
  1619. int nSamples;
  1620. if (!pVBIINFO) {
  1621. return NDSP_ERROR_ILLEGAL_VBIINFOHEADER;
  1622. }
  1623. nSamples= pVBIINFO->SamplesPerLine;
  1624. /* Actual start time must be less than or equal to desired start time */
  1625. /* This results in a positive offset into the input data */
  1626. nOffsetUS100= (DESIRED_START_TIME_US100 - pVBIINFO->ActualLineStartTime);
  1627. nOffsetData= nOffsetUS100*1000/NABTS_US_PER_CLOCK_TIMES_100000;
  1628. if (nOffsetData < 0) nOffsetData= 0;
  1629. if (nOffsetData + 1536 > nSamples) {
  1630. nOffsetData= nSamples-1536;
  1631. }
  1632. nSamples -= nOffsetData;
  1633. SASSERT(nSamples >= 1536);
  1634. *pnOffsetData= nOffsetData;
  1635. *pnOffsetSamples= nOffsetData;
  1636. return 0;
  1637. }
  1638. #endif
  1639. /*
  1640. Public function: also see nabtsapi.h.
  1641. This function is the key to the automatic equalization code.
  1642. Although its name suggests that it only does GCR-based equalization,
  1643. this function in fact coordinates all dynamic equalization, including
  1644. that done on the basis of the NABTS sync pattern. Therefore, be certain
  1645. to call this once per field even if you set g_bUseGCR to FALSE,
  1646. disabling GCR-based equalization and ghost cancellation.
  1647. If this function is never called, the DSP will simply use the
  1648. default fixed filter.
  1649. This function maintains the pState->nRetrainState value. This value
  1650. counts down each field, and determines the current state of the
  1651. equalization:
  1652. From g_nNabtsRetrainDelay down to 0: Idle, waiting to start retrain.
  1653. From 0 down to -g_nNabtsRetrainDuration: Acquire appropriate
  1654. GCR and NABTS sync signals
  1655. -g_nNabtsRetrainDuration: Perform equalization and ghost cancellation
  1656. based on acquired signal.
  1657. Go back to state g_nNabtsRetrainDelay afterwards.
  1658. Note that NDSPStartRetrain resets nRetrainState to zero, causing
  1659. immediate signal acquisition and retrain after g_nNabtsRetrainDuration
  1660. fields.
  1661. */
  1662. int NDSPProcessGCRLine(NDSPGCRStats *pLineStats,
  1663. unsigned char *pbSamples, NDSPState *pState,
  1664. int nFieldNumber, int nLineNumber,
  1665. KS_VBIINFOHEADER *pVBIINFO)
  1666. {
  1667. NDSPGCRStats gcrStats;
  1668. gcrStats.nSize= sizeof(gcrStats);
  1669. if (!pState || pState->uMagic != NDSP_STATE_MAGIC)
  1670. return NDSP_ERROR_ILLEGAL_NDSP_STATE;
  1671. pState->nRetrainState--;
  1672. gcrStats.bUsed= FALSE;
  1673. if (pState->nRetrainState < -g_nNabtsRetrainDuration) {
  1674. /* Signal acquired; perform equalization/ghost cancellation */
  1675. gcrStats.bUsed= TRUE;
  1676. DoEqualize(pState);
  1677. pState->nRetrainState= g_nNabtsRetrainDelay;
  1678. } else if (pState->nRetrainState < 0) {
  1679. /* Try to find best GCR line for later equalization */
  1680. /* NABTS sync is acquired by NDSPDecodeLine */
  1681. if (pState->nRetrainState == -1) {
  1682. /* Starting the search for a GCR line */
  1683. NDSPResetGCRAcquisition(pState);
  1684. NDSPResetNabsyncAcquisition(pState);
  1685. }
  1686. NDSPAcquireGCR(pState, pbSamples);
  1687. }
  1688. if (pLineStats) {
  1689. if (pLineStats->nSize < sizeof(NDSPGCRStats))
  1690. return NDSP_ERROR_ILLEGAL_STATS;
  1691. memcpy((void*)pLineStats, (void*)&gcrStats, sizeof(NDSPGCRStats));
  1692. }
  1693. return 0;
  1694. }
  1695. void NDSPResetGCRAcquisition(NDSPState *pState)
  1696. {
  1697. int i;
  1698. for (i= 0; i< NDSP_SIGS_TO_ACQUIRE; i++) {
  1699. pState->psmPosGCRs[i].dMaxval= 0.0;
  1700. pState->psmNegGCRs[i].dMaxval= 0.0;
  1701. }
  1702. }
  1703. void NDSPResetNabsyncAcquisition(NDSPState *pState)
  1704. {
  1705. int i;
  1706. for (i= 0; i< NDSP_SIGS_TO_ACQUIRE; i++) {
  1707. pState->psmSyncs[i].dMaxval= 0.0;
  1708. }
  1709. }
  1710. /* Average together multiple signals.
  1711. Use offsets computed by earlier convolution with target signal so
  1712. that we correctly align the signals before averaging */
  1713. /* 1 if error */
  1714. /* 0 if OK */
  1715. int NDSPAvgSigs(
  1716. NDSPState *pState, Double *pdDest, NDSPSigMatch *psm, Double *dConvRet)
  1717. {
  1718. int i,j;
  1719. Double dConv= 0.0;;
  1720. for (i= 0; i< pState->SamplesPerLine; i++) pdDest[i]= 0.0;
  1721. if (psm[NDSP_SIGS_TO_ACQUIRE-1].dMaxval == 0.0) return 1;
  1722. for (i= 0; i< pState->SamplesPerLine; i++) {
  1723. int nSum= 0;
  1724. for (j= 0; j< NDSP_SIGS_TO_ACQUIRE; j++) {
  1725. int from_index= i + psm[j].nOffset-20;;
  1726. if (from_index < 0) from_index= 0;
  1727. if (from_index > (pState->SamplesPerLine-1)) from_index= (pState->SamplesPerLine-1);
  1728. nSum += psm[j].pbSamples[from_index];
  1729. }
  1730. pdDest[i]= ((Double)nSum) / NDSP_SIGS_TO_ACQUIRE;
  1731. }
  1732. for (j= 0; j< NDSP_SIGS_TO_ACQUIRE; j++) {
  1733. dConv += psm[j].dMaxval;
  1734. }
  1735. dConv /= NDSP_SIGS_TO_ACQUIRE;
  1736. if (dConvRet) *dConvRet= dConv;
  1737. return 0;
  1738. }
  1739. /* We're trying to find the best representative matching signal during
  1740. the "Acquire appropriate GCR and NABTS sync signals" stage (see
  1741. NDSPProcessGCRLine).
  1742. Maintain an array of the N best signals see so far */
  1743. void NDSPTryToInsertSig(NDSPState *pState, NDSPSigMatch *psm, Double dMaxval,
  1744. unsigned char *pbSamples, int nOffset)
  1745. {
  1746. int i;
  1747. double dMaxvalToReplace= dMaxval;
  1748. int nIndexToReplace= -1;
  1749. for (i= 0; i< NDSP_SIGS_TO_ACQUIRE; i++) {
  1750. if (psm[i].dMaxval < dMaxvalToReplace) {
  1751. nIndexToReplace= i;
  1752. dMaxvalToReplace= psm[i].dMaxval;
  1753. }
  1754. }
  1755. if (nIndexToReplace >= 0) {
  1756. psm[nIndexToReplace].dMaxval= dMaxval;
  1757. psm[nIndexToReplace].nOffset= nOffset;
  1758. memcpy(psm[nIndexToReplace].pbSamples, pbSamples, pState->SamplesPerLine);
  1759. }
  1760. }
  1761. void NDSPAcquireNabsync(NDSPState *pState, unsigned char *pbSamples)
  1762. {
  1763. Double dMaxval = 0.0;
  1764. int nOffset = 0;
  1765. MatchWithEqualizeSignal(pState,
  1766. pbSamples,
  1767. &eqmatchNabtsSync,
  1768. &nOffset,
  1769. &dMaxval,
  1770. FALSE);
  1771. NDSPTryToInsertSig(pState, pState->psmSyncs, dMaxval, pbSamples, nOffset);
  1772. }
  1773. void NDSPAcquireGCR(NDSPState *pState, unsigned char *pbSamples)
  1774. {
  1775. Double dMaxval = 0.0;
  1776. int nOffset = 0;
  1777. MatchWithEqualizeSignal(pState,
  1778. pbSamples,
  1779. &eqmatchGCR1,
  1780. &nOffset,
  1781. &dMaxval,
  1782. TRUE);
  1783. if (dMaxval > 0.0) {
  1784. NDSPTryToInsertSig(pState, pState->psmPosGCRs, dMaxval, pbSamples, nOffset);
  1785. } else {
  1786. NDSPTryToInsertSig(pState, pState->psmNegGCRs, -dMaxval, pbSamples, nOffset);
  1787. }
  1788. }
  1789. /* This table is approximately given by:
  1790. 100 - (Chance of false positive / 3.2%) * 49.
  1791. However, we encode what would otherwise be all 100% with numbers close
  1792. to 100% to preserve some quality information, and what would otherwise
  1793. be off the scale in the negative direction, we encode with numbers close
  1794. to zero for the same reason */
  1795. int nConfidenceTable[25]= {
  1796. 100, /* 0 */
  1797. 99, /* 1 */
  1798. 95, /* 2 */
  1799. 90, /* 3 */
  1800. 75, /* 4 */
  1801. 49, /* 5 */
  1802. 45, /* 6 */
  1803. 40, /* 7 */
  1804. 16, /* 8 */
  1805. 15, /* 9 */
  1806. 14, /* 10 */
  1807. 13, /* 11 */
  1808. 12, /* 12 */
  1809. 11, /* 13 */
  1810. 10, /* 14 */
  1811. 9, /* 15 */
  1812. 8, /* 16 */
  1813. 7, /* 17 */
  1814. 6, /* 18 */
  1815. 5, /* 19 */
  1816. 4, /* 20 */
  1817. 3, /* 21 */
  1818. 2, /* 22 */
  1819. 1, /* 23 */
  1820. 0, /* 24 */
  1821. };
  1822. int NDSPSyncBytesToConfidence(unsigned char *pbSyncBytes)
  1823. {
  1824. int nReal=
  1825. (pbSyncBytes[0]<<16) +
  1826. (pbSyncBytes[1]<<8) +
  1827. (pbSyncBytes[2]<<0);
  1828. int nIdeal= 0x5555e7;
  1829. int nError= nReal ^ nIdeal;
  1830. int nErrors= 0;
  1831. /* Calculate the number of bit errors */
  1832. while (nError) {
  1833. nError &= (nError-1); /* Remove the least significant bit of nError */
  1834. nErrors++;
  1835. }
  1836. /* What are the chances that random data would match the sync bytes?
  1837. (2^24 / (24 choose nErrors))
  1838. #err % chance from random data
  1839. ---- -------------------------
  1840. 0 0.00001%
  1841. 1 0.00014%
  1842. 2 0.00165%
  1843. 3 0.01206%
  1844. 4 0.06334%
  1845. 5 0.25334%
  1846. 6 0.80225%
  1847. 7 2.06294%
  1848. 8 4.38375%
  1849. 9 7.79333%
  1850. 10 11.69000%
  1851. (These numbers aren't completely correct, since we found a spot
  1852. in the input line that convolves most closely with what we're expecting;
  1853. so the actual numbers depend a lot of the types of noise we get when
  1854. receiving non-NABTS lines.)
  1855. This code uses a cutoff of 4 bit errors. This was determined
  1856. empirically, because, in practice, the convolution finds a fair
  1857. number "reasonable" bits sometimes in random noise.
  1858. */
  1859. SASSERT(nErrors >= 0 && nErrors <= 24);
  1860. return nConfidenceTable[nErrors];
  1861. }
  1862. /* Some sample locations at which a "positive" GCR signal should be
  1863. quite positive */
  1864. /* ++++ Do these need to change on different sample rates? */
  1865. /* Compute confidence that we're looking at a GCR signal */
  1866. /* Do we think that pbSamples points at a GCR? Similar implementation
  1867. to NDSPSyncBytesToConfidence, above */
  1868. int NDSPDetectGCRConfidence(unsigned char *pbSamples)
  1869. {
  1870. int i;
  1871. int nTotalBits= 0;
  1872. int nGoodBits= 0;
  1873. int nCutoff= 30;
  1874. int nConfidence;
  1875. int nDC= float2long(Mean_8(pbSamples, g_nNabtsGcrSize * GCR_SAMPLE_RATE));
  1876. for (i= 0; i< sizeof(pnGCRPositiveIndices)/sizeof(int); i++) {
  1877. nTotalBits++;
  1878. nGoodBits += pbSamples[pnGCRPositiveIndices[i]] > nDC;
  1879. }
  1880. for (i= 0; i< sizeof(pnGCRNegativeIndices)/sizeof(int); i++) {
  1881. nTotalBits++;
  1882. nGoodBits += pbSamples[pnGCRNegativeIndices[i]] < nDC;
  1883. }
  1884. /* Cutoff is at 30 bits, based on statistics taken from samples. */
  1885. /* The actual confidence number here doesn't mean much except as
  1886. a diagnostic */
  1887. if (nGoodBits >= nCutoff) {
  1888. /* Good */
  1889. nConfidence= 51 + (nGoodBits - nCutoff) * 49 / (nTotalBits - nCutoff);
  1890. }
  1891. else {
  1892. /* Bad */
  1893. nConfidence= nGoodBits * 49 / nCutoff;
  1894. }
  1895. debug_printf(("GCR: %d good bits, %d total bits, conf %d\n",
  1896. nGoodBits, nTotalBits, nConfidence));
  1897. return nConfidence;
  1898. }
  1899. /* API for a possibly more efficient partial decode line.
  1900. This might be useful for determining group addresses before decoding
  1901. the rest of the line.
  1902. Currently, this API is no more efficient than the full
  1903. NDSPDecodeLine
  1904. Public function: also see nabtsapi.h */
  1905. int NDSPPartialDecodeLine(unsigned char *pbDest, NDSPLineStats *pLineStats,
  1906. unsigned char *pbSamples, NDSPState *pState,
  1907. int nFECType,
  1908. int nFieldNumber, int nLineNumber,
  1909. int nStart, int nEnd,
  1910. KS_VBIINFOHEADER *pVBIINFO)
  1911. {
  1912. debug_printf(("Warning: NDSPPartialDecodeLine uses slow implementation\n"));
  1913. return NDSPDecodeLine(pbDest, pLineStats,
  1914. pbSamples, pState,
  1915. nFECType,
  1916. nFieldNumber, nLineNumber,
  1917. pVBIINFO);
  1918. }
  1919. /* Main API for decoding a NABTS line.
  1920. *
  1921. * Inputs:
  1922. * pbSamples: pointer to 8-bit raw NABTS samples
  1923. * pState: NDSPState to use for decoding
  1924. * nFECType: Can be set to:
  1925. * NDSP_NO_FEC (don't use FEC information)
  1926. * NDSP_BUNDLE_FEC_1 (use Norpak-style bundle FEC info)
  1927. * NDSP_BUNDLE_FEC_2 (use Wavephore-style bundle FEC info)
  1928. * nFieldNumber:
  1929. * A number that increments by one for each successive field.
  1930. * "Odd" fields (as defined by NTSC) must be odd numbers
  1931. * "Even" fields must be even numbers.
  1932. * nLineNumber:
  1933. * The NTSC line (starting from the top of the field)
  1934. * from which this sample was taken.
  1935. *
  1936. * Outputs:
  1937. * pbDest: decoded data ("NABTS_BYTES_PER_LINE" (36) bytes long)
  1938. * pLineStats: stats on decoded data
  1939. *
  1940. * Errors:
  1941. *
  1942. * Returns 0 if no error
  1943. * Returns NDSP_ERROR_ILLEGAL_NDSP_STATE if state is illegal or uses
  1944. * unsupported settings
  1945. * Returns NDSP_ERROR_ILLEGAL_STATS if stats is passed incorrectly
  1946. *
  1947. * Notes:
  1948. * pbDest must point to a buffer at least 36 bytes long
  1949. * pLineStats->nSize must be set to sizeof(*pLineStats) prior to call
  1950. * (to maintain backwards compatibility should we add fields in the future)
  1951. * pLineStats->nSize will be set to the size actually filled in by
  1952. * the call (the number will stay the same or get smaller)
  1953. * Currently the routine only supports a pState with an FIR filter
  1954. * that has 5 taps
  1955. * nFECType is currently unused, but would potentially be used to give
  1956. * FEC feedback to the DSP decode, for possible tuning and/or retry
  1957. Public function: also see nabtsapi.h
  1958. */
  1959. #ifdef DEBUG
  1960. extern void BPCplot(unsigned char *buf, unsigned long offs, unsigned long len);
  1961. extern void BPCplotInd(unsigned char *buf, unsigned long offs, unsigned long len, long index);
  1962. unsigned short NDSPplotSync = 0;
  1963. unsigned short NDSPplotBreak = 0;
  1964. unsigned short NDSPplotLen = 75;
  1965. unsigned short NDSPplotBits = 0;
  1966. unsigned short NDSPplotBitsSkip = 0;
  1967. #endif //DEBUG
  1968. int NDSPDecodeLine(unsigned char *pbDest, NDSPLineStats *pLineStats,
  1969. unsigned char *pbSamples, NDSPState *pState,
  1970. int nFECType,
  1971. int nFieldNumber, int nLineNumber,
  1972. KS_VBIINFOHEADER *pVBIINFO)
  1973. {
  1974. Double dSyncStart;
  1975. Double dConvConfidence;
  1976. int nConfidence;
  1977. Double dDC, dAC;
  1978. int ret;
  1979. if (!pState) return NDSP_ERROR_ILLEGAL_NDSP_STATE;
  1980. /* These args aren't yet used */
  1981. (void)nFieldNumber;
  1982. (void)nLineNumber;
  1983. /* Save number of samples */
  1984. pState->SamplesPerLine = pVBIINFO->SamplesPerLine;
  1985. /* Locate the NABTS sync so that we know the correct
  1986. offset from which to decode the signal */
  1987. NDSPDecodeFindSync(pbSamples, &dConvConfidence,
  1988. &dSyncStart, &dDC, &dAC, pState, pVBIINFO);
  1989. #ifdef DEBUG
  1990. if (NDSPplotSync)
  1991. {
  1992. unsigned long lSyncStart;
  1993. long index;
  1994. lSyncStart = float2long(dSyncStart);
  1995. DbgPrint("NDSPDecodeLine: SyncStart at %s\n", flPrintf(dSyncStart, 4));
  1996. index = lSyncStart - NDSPplotLen / 2;
  1997. if (index < 0)
  1998. index = 0;
  1999. BPCplotInd(pbSamples, index, NDSPplotLen, lSyncStart - index);
  2000. if (NDSPplotBreak)
  2001. debug_breakpoint();
  2002. }
  2003. #endif //DEBUG
  2004. /* Perform the DSP decode */
  2005. ret= NDSPGetBits(pbDest, pbSamples, dSyncStart, dDC, pState, nFECType);
  2006. /* Do the sync bytes look reasonable? */
  2007. nConfidence= NDSPSyncBytesToConfidence(pbDest);
  2008. if (nConfidence >= 50 &&
  2009. pState->nRetrainState < 0) {
  2010. /* If the sync bytes look reasonable, and we're in the middle
  2011. of the retrain stage, possibly store this signal away for
  2012. later equalization.
  2013. (See NDSPProcessGCRLine for more information) */
  2014. NDSPAcquireNabsync(pState, pbSamples);
  2015. }
  2016. #ifdef DEBUG_VERBOSE
  2017. if (g_bDebugSyncBytes) {
  2018. debug_printf(("Sync found at %s with conv=%s, conf= %d, DC=%s, AC=%s\n",
  2019. flPrintf(dSyncStart, 2),
  2020. flPrintf(dConvConfidence, 6),
  2021. nConfidence,
  2022. flPrintf(dDC, 6),
  2023. flPrintf(dAC, 6)));
  2024. }
  2025. #endif
  2026. if (ret != 0) return ret;
  2027. #ifdef DEBUG_VERBOSE
  2028. if (g_bDebugSyncBytes) {
  2029. debug_printf(("Sync bytes %x %x %x\n", pbDest[0], pbDest[1], pbDest[2]));
  2030. }
  2031. #endif
  2032. /* Fill in the line stats structure */
  2033. if (pLineStats) {
  2034. /* FUTURE: If we add fields, we'll need to change this logic */
  2035. if (pLineStats->nSize < sizeof(NDSPLineStats))
  2036. return NDSP_ERROR_ILLEGAL_STATS;
  2037. pLineStats->nSize= sizeof(pLineStats);
  2038. pLineStats->nConfidence= nConfidence;
  2039. pLineStats->nFirstBit= float2long(dSyncStart);
  2040. pLineStats->dDC= (double) dDC;
  2041. }
  2042. return 0;
  2043. }
  2044. /*********************/
  2045. /* PRIVATE FUNCTIONS */
  2046. /*********************/
  2047. /* Set the filter for the given DSP state */
  2048. int NDSPStateSetFilter(NDSPState *pState, FIRFilter *filter, int nFilterSize)
  2049. {
  2050. if (nFilterSize != sizeof(FIRFilter) ||
  2051. filter->nTaps < 1 ||
  2052. filter->nTaps > NDSP_MAX_FIR_COEFFS) {
  2053. return NDSP_ERROR_ILLEGAL_FILTER;
  2054. }
  2055. if (!filter->bVariableTaps) {
  2056. if ( !(filter->nTaps == 21 && filter->dTapSpacing == 2)
  2057. && !(filter->nTaps == 5 && filter->dTapSpacing == 5)
  2058. && !(filter->nTaps == 5 && filter->dTapSpacing == 4)
  2059. && !(filter->nTaps == 5 && fltCmp(filter->dTapSpacing, KS_47NABTS_SCALER)) )
  2060. /* ===== add cases as needed ===== */
  2061. return NDSP_ERROR_ILLEGAL_FILTER;
  2062. }
  2063. // NormalizeFilter(&g_filterDefault);
  2064. NormalizeFilter(filter);
  2065. memcpy((void*) &pState->filter, (void*) filter, sizeof(FIRFilter));
  2066. return 0;
  2067. }
  2068. /* Verify that the "DC" content of the filter is 1.
  2069. That is, the taps of the filter should add to 1.
  2070. This way, when we convolve the filter with a signal, we don't
  2071. change the DC content of the signal */
  2072. void NormalizeFilter(FIRFilter *pFilter)
  2073. {
  2074. Double dFactor= 1.0 / Sum_d(pFilter->pdTaps, pFilter->nTaps);
  2075. Mult_d(pFilter->pdTaps, pFilter->pdTaps, pFilter->nTaps, dFactor);
  2076. }
  2077. /* Find the NABTS sync signal.
  2078. Algorithm:
  2079. Convolve a window around where the sync is expected with the actual
  2080. transmitted sync waveform. This will give a very robust location
  2081. for the sync (including the exact phase for best sampling).
  2082. The typical disadvantage to convolution is that it is slow, but,
  2083. for this particular waveform, it was possible to hardcode an
  2084. algorithm that takes approximately 11 additive operations per
  2085. location tested, rather than a more typical 24 multiplies and 72
  2086. additions per location tested.
  2087. # of additive operations ~=
  2088. (11 * (g_nNabtsDecodeBeginMax - g_nNabtsDecodeBeginMin))
  2089. The smaller the window between g_nNabtsDecodeBeginMin and
  2090. g_nNabtsDecodeBeginMax, the faster this routine will run.
  2091. */
  2092. #ifdef DEBUG
  2093. int DSPsyncShowFineTune = 0;
  2094. #endif //DEBUG
  2095. void NDSPDecodeFindSync(unsigned char *pbSamples,
  2096. Double *pdSyncConfidence,
  2097. Double *pdSyncStart, Double *pdDC, Double *pdAC,
  2098. NDSPState* pState, KS_VBIINFOHEADER *pVBIINFO)
  2099. {
  2100. Double dBestMatch= -1000000;
  2101. Double dBestPos= 0;
  2102. Double dStart;
  2103. /* compute byte-based offsets from time-based offsets: */
  2104. #define NABTS_BITS_SEC (1.0/(double)NABTS_RATE)
  2105. #define BEGIN_MIN_TIME (9.36e-6 - (pVBIINFO->ActualLineStartTime/1e8))
  2106. #define BEGIN_MAX_TIME (11.63e-6 - (pVBIINFO->ActualLineStartTime/1e8))
  2107. /* some shorthand... */
  2108. #define ITR_AR(arr, idx) _InterpUCharArr(arr, idx)
  2109. Double dBegin_min_offset = BEGIN_MIN_TIME * (NABTS_RATE * pState->dSampleRate);
  2110. Double dBegin_max_offset = BEGIN_MAX_TIME * (NABTS_RATE * pState->dSampleRate);
  2111. /* +++++++ check that dBegin_min_offset >= 0, and
  2112. dBegin_max_offset is not so high that decoding will go off the end
  2113. of the array */
  2114. Double dSamp = pState->dSampleRate;
  2115. for (dStart= dBegin_min_offset;
  2116. dStart< dBegin_min_offset+dSamp;
  2117. dStart += 1) {
  2118. Double d= dStart;
  2119. Double dMatch;
  2120. Double dMatch1=
  2121. ITR_AR(pbSamples,d+ 0*dSamp) - ITR_AR(pbSamples,d+ 1*dSamp) /* 0-1 */
  2122. + ITR_AR(pbSamples,d+ 2*dSamp) - ITR_AR(pbSamples,d+ 3*dSamp) /* 2-3 */
  2123. + ITR_AR(pbSamples,d+ 4*dSamp) - ITR_AR(pbSamples,d+ 5*dSamp) /* 4-5 */
  2124. + ITR_AR(pbSamples,d+ 6*dSamp) - ITR_AR(pbSamples,d+ 7*dSamp) /* 6-7 */
  2125. + ITR_AR(pbSamples,d+ 8*dSamp) - ITR_AR(pbSamples,d+ 9*dSamp) /* 8-9 */
  2126. + ITR_AR(pbSamples,d+10*dSamp) - ITR_AR(pbSamples,d+11*dSamp) /* 10-11 */
  2127. + ITR_AR(pbSamples,d+12*dSamp) - ITR_AR(pbSamples,d+13*dSamp) /* 12-13 */
  2128. + ITR_AR(pbSamples,d+14*dSamp) - ITR_AR(pbSamples,d+15*dSamp);/* 14-15 */
  2129. Double dMatch2=
  2130. ITR_AR(pbSamples,d+16*dSamp)
  2131. + ITR_AR(pbSamples,d+17*dSamp)
  2132. + ITR_AR(pbSamples,d+18*dSamp) /* 0-2 */
  2133. - ITR_AR(pbSamples,d+19*dSamp)
  2134. - ITR_AR(pbSamples,d+20*dSamp); /* 3-4 */
  2135. /* Note that we skip bits 5-7 of the third byte because they're not
  2136. necessary for locating the sync, and this speeds us up a bit */
  2137. while (d < dBegin_max_offset) {
  2138. dMatch= dMatch1 + dMatch2;
  2139. if (dMatch > dBestMatch) {
  2140. dBestMatch= dMatch;
  2141. dBestPos= d;
  2142. }
  2143. /* Move dMatch1 forward by dSamp */
  2144. dMatch1 -= _InterpUCharArr(pbSamples,d+0*dSamp);
  2145. dMatch1 += _InterpUCharArr(pbSamples,d+16*dSamp);
  2146. dMatch1 = (-dMatch1);
  2147. /* Move dMatch2 forward by dSamp */
  2148. dMatch2 -= _InterpUCharArr(pbSamples,d+21*dSamp);
  2149. dMatch2 += (2 * _InterpUCharArr(pbSamples,d+19*dSamp));
  2150. dMatch2 -= _InterpUCharArr(pbSamples,d+16*dSamp);
  2151. d += dSamp;
  2152. }
  2153. }
  2154. #ifdef FINETUNE
  2155. // Now try to fine-tune dBestPos by choosing the offset with the maximum
  2156. // five-bit amplitude sum
  2157. {
  2158. Double d, dNewBest, dPeakSum, dNewSum;
  2159. // Start by choosing dBestPos as the 'best'
  2160. dNewBest =
  2161. d = dBestPos;
  2162. dPeakSum =
  2163. /*ITR_AR(pbSamples,d+ 0*dSamp) - ITR_AR(pbSamples,d+ 1*dSamp) /* 0-1 */
  2164. + ITR_AR(pbSamples,d+ 2*dSamp) - ITR_AR(pbSamples,d+ 3*dSamp) /* 2-3 */
  2165. + ITR_AR(pbSamples,d+ 4*dSamp) - ITR_AR(pbSamples,d+ 5*dSamp) /* 4-5 */
  2166. + ITR_AR(pbSamples,d+ 6*dSamp) - ITR_AR(pbSamples,d+ 7*dSamp) /* 6-7 */
  2167. + ITR_AR(pbSamples,d+ 8*dSamp) - ITR_AR(pbSamples,d+ 9*dSamp) /* 8-9 */
  2168. + ITR_AR(pbSamples,d+10*dSamp) - ITR_AR(pbSamples,d+11*dSamp) /* 10-11 */
  2169. + ITR_AR(pbSamples,d+12*dSamp) - ITR_AR(pbSamples,d+13*dSamp) /* 12-13 */
  2170. + ITR_AR(pbSamples,d+14*dSamp) - ITR_AR(pbSamples,d+15*dSamp) /* 14-15 */
  2171. #if 0
  2172. + ITR_AR(pbSamples,d+16*dSamp)
  2173. + ITR_AR(pbSamples,d+17*dSamp)
  2174. + ITR_AR(pbSamples,d+18*dSamp) /* 0-2 */
  2175. - ITR_AR(pbSamples,d+19*dSamp)
  2176. - ITR_AR(pbSamples,d+20*dSamp) /* 3-4 */
  2177. #endif
  2178. ;
  2179. for (d = dBestPos - 1; d < dBestPos + 2; d += 0.1) {
  2180. dNewSum =
  2181. /*ITR_AR(pbSamples,d+ 0*dSamp) - ITR_AR(pbSamples,d+ 1*dSamp); /* 0-1 */
  2182. + ITR_AR(pbSamples,d+ 2*dSamp) - ITR_AR(pbSamples,d+ 3*dSamp) /* 2-3 */
  2183. + ITR_AR(pbSamples,d+ 4*dSamp) - ITR_AR(pbSamples,d+ 5*dSamp) /* 4-5 */
  2184. + ITR_AR(pbSamples,d+ 6*dSamp) - ITR_AR(pbSamples,d+ 7*dSamp) /* 6-7 */
  2185. + ITR_AR(pbSamples,d+ 8*dSamp) - ITR_AR(pbSamples,d+ 9*dSamp) /* 8-9 */
  2186. + ITR_AR(pbSamples,d+10*dSamp) - ITR_AR(pbSamples,d+11*dSamp) /* 10-11 */
  2187. + ITR_AR(pbSamples,d+12*dSamp) - ITR_AR(pbSamples,d+13*dSamp) /* 12-13 */
  2188. + ITR_AR(pbSamples,d+14*dSamp) - ITR_AR(pbSamples,d+15*dSamp) /* 14-15 */
  2189. #if 0
  2190. + ITR_AR(pbSamples,d+16*dSamp)
  2191. + ITR_AR(pbSamples,d+17*dSamp)
  2192. + ITR_AR(pbSamples,d+18*dSamp) /* 0-2 */
  2193. - ITR_AR(pbSamples,d+19*dSamp)
  2194. - ITR_AR(pbSamples,d+20*dSamp) /* 3-4 */
  2195. #endif
  2196. ;
  2197. if (dNewSum > dPeakSum) {
  2198. dPeakSum = dNewSum;
  2199. dNewBest = d;
  2200. }
  2201. }
  2202. if (dNewBest != dBestPos) {
  2203. if (DSPsyncShowFineTune)
  2204. debug_printf(("NDSPDecodeFindSync: fine-tuned dBestPos from %s to %s\n", flPrintf(dBestPos,6), flPrintf(dNewBest,6)));
  2205. dBestPos = dNewBest;
  2206. }
  2207. }
  2208. #endif //FINETUNE
  2209. *pdSyncStart= dBestPos;
  2210. {
  2211. /* Calculate DC offset */
  2212. Double dDC= 0, dAC= 0;
  2213. Double d;
  2214. for (d= dBestPos; d < dBestPos + (16*dSamp); d += dSamp) {
  2215. dDC += _InterpUCharArr(pbSamples,d);
  2216. }
  2217. dDC /= 16.0;
  2218. /* Calculate AC amplitude */
  2219. for (d= dBestPos; d < dBestPos + (16*dSamp); d += dSamp) {
  2220. Double tmp = _InterpUCharArr(pbSamples,d)-dDC;
  2221. dAC += (tmp < 0)? -tmp : tmp;
  2222. }
  2223. dAC /= 16.0;
  2224. if (dAC < 1.0) dAC= 1.0; /* Prevent divide by 0 later */
  2225. *pdDC= dDC;
  2226. *pdAC= dAC;
  2227. /* Confidence is from 0 to approximately 96 (4 * 24) */
  2228. /* Subtract out 1*nDC since we had one more high bit than low
  2229. in our convolution */
  2230. *pdSyncConfidence= 2 * (dBestMatch - dDC) / dAC;
  2231. /* If dAC < 20, reduce the confidence */
  2232. *pdSyncConfidence = (*pdSyncConfidence * dAC / 20);
  2233. }
  2234. }
  2235. /* Perform the DSP to get the NABTS bits from the raw sample line.
  2236. The basic algorithm is to convolve the input signal with the current
  2237. filter, and detect for each bit whether the bit is above or below the
  2238. DC level as determined by looking at the NABTS sync, which is a known
  2239. signal.
  2240. This is where we spend most of our cycles, so there are lots of
  2241. specially-cased hard-coded routines to handle the different filter
  2242. configurations we currently use.
  2243. */
  2244. int NDSPGetBits(unsigned char *pbDest, unsigned char *pbSamples,
  2245. Double dSyncStart, double dDC, NDSPState *pState,
  2246. int nFECType)
  2247. {
  2248. int ret= 0;
  2249. if (pState->filter.bVariableTaps) {
  2250. /* Variable-tap filter */
  2251. ret= NDSPGetBits_var(pbDest, pbSamples, float2long(dSyncStart), dDC,
  2252. pState, nFECType);
  2253. } else {
  2254. /* change to multiple else if's */
  2255. if ( pState->filter.nTaps == 21 && pState->filter.dTapSpacing == 2) {
  2256. /* Fixed taps {-20, -18, ... 18, 20} */
  2257. ret= NDSPGetBits_21_2(pbDest, pbSamples, float2long(dSyncStart),
  2258. dDC, pState, nFECType);
  2259. }
  2260. else if ( pState->filter.nTaps == 5 && fltCmp(pState->filter.dTapSpacing, 5)) {
  2261. /* Fixed taps {-10, -5, 0, 5, 10} */
  2262. ret= NDSPGetBits_5_5(pbDest, pbSamples, float2long(dSyncStart), dDC, pState, nFECType);
  2263. }
  2264. else if ( pState->filter.nTaps == 5 && fltCmp(pState->filter.dTapSpacing, 4)) {
  2265. ret= NDSPGetBits_5_4(pbDest, pbSamples, float2long(dSyncStart), dDC, pState, nFECType);
  2266. }
  2267. else if ( pState->filter.nTaps == 5 && fltCmp(pState->filter.dTapSpacing,KS_47NABTS_SCALER) ) {
  2268. ret= NDSPGetBits_5_47(pbDest, pbSamples, dSyncStart, dDC, pState, nFECType);
  2269. }
  2270. else {
  2271. ret= NDSP_ERROR_ILLEGAL_NDSP_STATE;
  2272. }
  2273. }
  2274. return ret;
  2275. }
  2276. /* Get bits for fixed taps {-10, -5, 0, 5, 10}
  2277. See NDSPGetBits for more information */
  2278. int NDSPGetBits_5_5(unsigned char *pbDest, unsigned char *pbSamples,
  2279. int nSyncStart, double dDC, NDSPState *pState,
  2280. int nFECType)
  2281. {
  2282. int nSamplePos= nSyncStart;
  2283. int i;
  2284. Double a,b,c,d,e;
  2285. #ifdef UseDoubleCoeffs
  2286. Double dCoeffA= pState->filter.pdTaps[0];
  2287. Double dCoeffB= pState->filter.pdTaps[1];
  2288. Double dCoeffC= pState->filter.pdTaps[2];
  2289. Double dCoeffD= pState->filter.pdTaps[3];
  2290. Double dCoeffE= pState->filter.pdTaps[4];
  2291. dDC *= (dCoeffA + dCoeffB + dCoeffC + dCoeffD + dCoeffE);
  2292. #else //UseDoubleCoeffs
  2293. Double *pdTaps = pState->filter.pdTaps;
  2294. int dCoeffA= float2long(pdTaps[0]);
  2295. int dCoeffB= float2long(pdTaps[1]);
  2296. int dCoeffC= float2long(pdTaps[2]);
  2297. int dCoeffD= float2long(pdTaps[3]);
  2298. int dCoeffE= float2long(pdTaps[4]);
  2299. dDC *= (dCoeffA + dCoeffB + dCoeffC + dCoeffD + dCoeffE);
  2300. //dDC *= (pdTaps[0] + pdTaps[1] + pdTaps[2] + pdTaps[3] + pdTaps[4]);
  2301. #endif //UseDoubleCoeffs
  2302. (void)nFECType;
  2303. SASSERT(pState);
  2304. SASSERT(pState->filter.nTaps == 5);
  2305. SASSERT(pState->filter.dTapSpacing == 5);
  2306. dDC *= (dCoeffA + dCoeffB + dCoeffC + dCoeffD + dCoeffE);
  2307. if (nSamplePos-10 >= 0) {
  2308. a= (Double) pbSamples[nSamplePos-10];
  2309. } else {
  2310. a= 0.0;
  2311. }
  2312. if (nSamplePos-5 >= 0) {
  2313. b= (Double) pbSamples[nSamplePos- 5];
  2314. } else {
  2315. b= 0.0;
  2316. }
  2317. c= (Double) pbSamples[nSamplePos+ 0];
  2318. d= (Double) pbSamples[nSamplePos+ 5];
  2319. e= (Double) pbSamples[nSamplePos+10];
  2320. #ifdef DEBUG
  2321. NDSPplotBitsSkip = 0;
  2322. #endif //DEBUG
  2323. for (i= 0; i< NABTS_BYTES_PER_LINE; i++) {
  2324. int j;
  2325. int nByte= 0;
  2326. for (j= 0; j< 8; j++) {
  2327. #ifdef DEBUG
  2328. int bit = 0;
  2329. #endif //DEBUG
  2330. nByte >>= 1;
  2331. if (a*dCoeffA + b*dCoeffB + c*dCoeffC + d*dCoeffD + e*dCoeffE > dDC) {
  2332. nByte |= 128;
  2333. #ifdef DEBUG
  2334. bit = 1;
  2335. #endif //DEBUG
  2336. }
  2337. #ifdef DEBUG
  2338. if (NDSPplotBits && !NDSPplotBitsSkip)
  2339. {
  2340. long index;
  2341. DbgPrint("NDSPGetBits_5_5: nSamplePos = %d, dDC = %s, bit = %d\n",
  2342. nSamplePos, flPrintf(dDC, 4), bit);
  2343. index = nSamplePos - NDSPplotLen / 2;
  2344. if (index < 0)
  2345. index = 0;
  2346. if (index + NDSPplotLen > pState->SamplesPerLine)
  2347. index = pState->SamplesPerLine - NDSPplotLen;
  2348. BPCplotInd(pbSamples, index, NDSPplotLen, nSamplePos - index);
  2349. if (NDSPplotBreak)
  2350. debug_breakpoint();
  2351. }
  2352. #endif //DEBUG
  2353. a=b;
  2354. b=c;
  2355. c=d;
  2356. d=e;
  2357. nSamplePos += 5;
  2358. if (nSamplePos+10 < pState->SamplesPerLine ) {
  2359. e= (Double) pbSamples[nSamplePos+10];
  2360. } else {
  2361. e= 0.0;
  2362. }
  2363. }
  2364. pbDest[i]= nByte;
  2365. }
  2366. return 0;
  2367. }
  2368. /* Get bits for fixed taps {-9.428, -4.714, 0, 4.714, 9.428} //FIXME
  2369. See NDSPGetBits for more information */
  2370. int NDSPGetBits_5_47(unsigned char *pbDest, unsigned char *pbSamples,
  2371. Double dSyncStart, double dDC, NDSPState *pState,
  2372. int nFECType)
  2373. {
  2374. double dSamplePos= dSyncStart;
  2375. int i;
  2376. Double dSpacing;
  2377. #ifdef UseMultiTaps
  2378. Double a,b,c,d,e;
  2379. # ifdef UseDoubleCoeffs
  2380. Double dCoeffA= pState->filter.pdTaps[0];
  2381. Double dCoeffB= pState->filter.pdTaps[1];
  2382. Double dCoeffC= pState->filter.pdTaps[2];
  2383. Double dCoeffD= pState->filter.pdTaps[3];
  2384. Double dCoeffE= pState->filter.pdTaps[4];
  2385. dDC *= (dCoeffA + dCoeffB + dCoeffC + dCoeffD + dCoeffE);
  2386. # else //UseDoubleCoeffs
  2387. Double *pdTaps = pState->filter.pdTaps;
  2388. int dCoeffA= float2long(pdTaps[0]);
  2389. int dCoeffB= float2long(pdTaps[1]);
  2390. int dCoeffC= float2long(pdTaps[2]);
  2391. int dCoeffD= float2long(pdTaps[3]);
  2392. int dCoeffE= float2long(pdTaps[4]);
  2393. dDC *= (dCoeffA + dCoeffB + dCoeffC + dCoeffD + dCoeffE);
  2394. //dDC *= (pdTaps[0] + pdTaps[1] + pdTaps[2] + pdTaps[3] + pdTaps[4]);
  2395. # endif //UseDoubleCoeffs
  2396. #else //UseMultiTaps
  2397. #endif //UseMultiTaps
  2398. (void)nFECType;
  2399. SASSERT(pState);
  2400. SASSERT(pState->filter.nTaps == 5);
  2401. SASSERT(fltCmp(pState->filter.dTapSpacing, KS_47NABTS_SCALER));
  2402. dSpacing = pState->filter.dTapSpacing;
  2403. #ifdef UseMultiTaps
  2404. if (dSamplePos-(2*KS_47NABTS_SCALER) >= 0) {
  2405. a= (Double) _InterpUCharArr(pbSamples, dSamplePos-2*dSpacing);
  2406. } else {
  2407. a= 0.0;
  2408. }
  2409. if (dSamplePos-KS_47NABTS_SCALER >= 0) {
  2410. b= (Double) _InterpUCharArr(pbSamples, dSamplePos-dSpacing);
  2411. } else {
  2412. b= 0.0;
  2413. }
  2414. c= (Double) _InterpUCharArr(pbSamples, dSamplePos + 0 );
  2415. d= (Double) _InterpUCharArr(pbSamples, dSamplePos + dSpacing );
  2416. e= (Double) _InterpUCharArr(pbSamples, dSamplePos + 2*dSpacing );
  2417. #endif //UseMultiTaps
  2418. #ifdef DEBUG
  2419. NDSPplotBitsSkip = 0;
  2420. #endif //DEBUG
  2421. for (i= 0; i< NABTS_BYTES_PER_LINE; i++) {
  2422. int j;
  2423. int nByte= 0;
  2424. for (j= 0; j< 8; j++) {
  2425. #ifdef DEBUG
  2426. int bit = 0;
  2427. #endif //DEBUG
  2428. nByte >>= 1;
  2429. #ifdef UseMultiTaps
  2430. if (a*dCoeffA + b*dCoeffB + c*dCoeffC + d*dCoeffD + e*dCoeffE > dDC)
  2431. #else //UseMultiTaps
  2432. if (_InterpUCharArr(pbSamples, dSamplePos) > dDC)
  2433. #endif //UseMultiTaps
  2434. {
  2435. nByte |= 128;
  2436. #ifdef DEBUG
  2437. bit = 1;
  2438. #endif //DEBUG
  2439. }
  2440. #ifdef DEBUG
  2441. if (NDSPplotBits && !NDSPplotBitsSkip)
  2442. {
  2443. unsigned long lSPint;
  2444. long index;
  2445. lSPint = float2long(dSamplePos);
  2446. DbgPrint("NDSPGetBits_5_47: dSamplePos = %s, dDC = %s, bit = %d\n",
  2447. flPrintf(dSamplePos, 4), flPrintf(dDC, 4), bit);
  2448. index -= lSPint - NDSPplotLen / 2;
  2449. if (index < 0)
  2450. index = 0;
  2451. BPCplotInd(pbSamples, index, NDSPplotLen, lSPint - index);
  2452. if (NDSPplotBreak)
  2453. debug_breakpoint();
  2454. }
  2455. #endif //DEBUG
  2456. dSamplePos += dSpacing;
  2457. #ifdef UseMultiTaps
  2458. a=b;
  2459. b=c;
  2460. c=d;
  2461. d=e;
  2462. if ( dSamplePos+(2*dSpacing) < pState->SamplesPerLine ) {
  2463. e= _InterpUCharArr(pbSamples, dSamplePos+2*dSpacing);
  2464. } else {
  2465. e= 0.0;
  2466. }
  2467. #endif //UseMultiTaps
  2468. }
  2469. pbDest[i]= nByte;
  2470. }
  2471. return 0;
  2472. }
  2473. /* Get bits for fixed taps {-8, -4, 0, 4, 8}
  2474. See NDSPGetBits for more information */
  2475. int NDSPGetBits_5_4(unsigned char *pbDest, unsigned char *pbSamples,
  2476. int nSyncStart, double dDC, NDSPState *pState,
  2477. int nFECType)
  2478. {
  2479. int nSamplePos= nSyncStart;
  2480. int i;
  2481. Double a,b,c,d,e;
  2482. #ifdef UseDoubleCoeffs
  2483. Double dCoeffA= pState->filter.pdTaps[0];
  2484. Double dCoeffB= pState->filter.pdTaps[1];
  2485. Double dCoeffC= pState->filter.pdTaps[2];
  2486. Double dCoeffD= pState->filter.pdTaps[3];
  2487. Double dCoeffE= pState->filter.pdTaps[4];
  2488. dDC *= (dCoeffA + dCoeffB + dCoeffC + dCoeffD + dCoeffE);
  2489. #else //UseDoubleCoeffs
  2490. Double *pdTaps = pState->filter.pdTaps;
  2491. int dCoeffA= float2long(pdTaps[0]);
  2492. int dCoeffB= float2long(pdTaps[1]);
  2493. int dCoeffC= float2long(pdTaps[2]);
  2494. int dCoeffD= float2long(pdTaps[3]);
  2495. int dCoeffE= float2long(pdTaps[4]);
  2496. dDC *= (dCoeffA + dCoeffB + dCoeffC + dCoeffD + dCoeffE);
  2497. //dDC *= (pdTaps[0] + pdTaps[1] + pdTaps[2] + pdTaps[3] + pdTaps[4]);
  2498. #endif //UseDoubleCoeffs
  2499. (void)nFECType;
  2500. SASSERT(pState);
  2501. SASSERT(pState->filter.nTaps == 5);
  2502. SASSERT(pState->filter.dTapSpacing == 4);
  2503. if (nSamplePos-8 >= 0) {
  2504. a= (Double) pbSamples[nSamplePos-8];
  2505. } else {
  2506. a= 0.0;
  2507. }
  2508. if (nSamplePos-4 >= 0) {
  2509. b= (Double) pbSamples[nSamplePos- 4];
  2510. } else {
  2511. b= 0.0;
  2512. }
  2513. c= (Double) pbSamples[nSamplePos+ 0];
  2514. d= (Double) pbSamples[nSamplePos+ 4];
  2515. e= (Double) pbSamples[nSamplePos+8];
  2516. for (i= 0; i< NABTS_BYTES_PER_LINE; i++) {
  2517. int j;
  2518. int nByte= 0;
  2519. for (j= 0; j< 8; j++) {
  2520. nByte >>= 1;
  2521. if (a*dCoeffA + b*dCoeffB + c*dCoeffC + d*dCoeffD + e*dCoeffE
  2522. > dDC) nByte |= 128;
  2523. a=b;
  2524. b=c;
  2525. c=d;
  2526. d=e;
  2527. nSamplePos += 4;
  2528. if (nSamplePos+8 < pState->SamplesPerLine ) {
  2529. e= (Double) pbSamples[nSamplePos+8];
  2530. } else {
  2531. e= 0.0;
  2532. }
  2533. }
  2534. pbDest[i]= nByte;
  2535. }
  2536. return 0;
  2537. }
  2538. #if 0
  2539. int NDSPGetBits_no_filter(unsigned char *pbDest, unsigned char *pbSamples,
  2540. int nSyncStart, double dDC, NDSPState *pState,
  2541. int nFECType)
  2542. {
  2543. int nSamplePos= nSyncStart;
  2544. int i;
  2545. (void)nFECType;
  2546. SASSERT(pState);
  2547. for (i= 0; i< NABTS_BYTES_PER_LINE; i++) {
  2548. int j;
  2549. int nByte= 0;
  2550. for (j= 0; j< 8; j++) {
  2551. nByte >>= 1;
  2552. if (pbSamples[nSamplePos] > dDC) nByte |= 128;
  2553. nSamplePos += float2long(pState->dSampleRate); // dSR should be made non-floating point.
  2554. }
  2555. pbDest[i]= nByte;
  2556. }
  2557. return 0;
  2558. }
  2559. #endif //0
  2560. Double Sum_d(Double *a, int a_size)
  2561. {
  2562. Double sum= 0.0;
  2563. int i;
  2564. SASSERT(a_size>=0);
  2565. for (i= 0; i< a_size; i++) sum += a[i];
  2566. return sum;
  2567. }
  2568. /* Get bits for fixed taps {-20, -18, ... 18, 20}
  2569. See NDSPGetBits for more information */
  2570. int NDSPGetBits_21_2(unsigned char *pbDest, unsigned char *pbSamples,
  2571. int nSyncStart, double dDC, NDSPState *pState,
  2572. int nFECType)
  2573. {
  2574. int nSamplePos= nSyncStart;
  2575. int i;
  2576. Double *pdLine;
  2577. Double *pdTaps;
  2578. Double dDCa, dDCb;
  2579. int nTaps;
  2580. int nTapSpacing;
  2581. int nTail;
  2582. int index;
  2583. unsigned char pbBuf[NABTS_BYTES_PER_LINE];
  2584. /*unsigned char pbBufa[NABTS_BYTES_PER_LINE];*/
  2585. /*unsigned char pbBufb[NABTS_BYTES_PER_LINE];*/
  2586. SASSERT(pState);
  2587. SASSERT(pState->filter.nTaps == 21);
  2588. SASSERT(pState->filter.dTapSpacing == 2);
  2589. SASSERT(!pState->bUsingScratch1);
  2590. pState->bUsingScratch1= __LINE__;
  2591. pdLine= pState->pdScratch1;
  2592. SASSERT(sizeof(pState->pdScratch1)/sizeof(Double) >=
  2593. pState->SamplesPerLine+(21-1)*2);
  2594. pdTaps= pState->filter.pdTaps;
  2595. nTaps= pState->filter.nTaps;
  2596. nTapSpacing= float2long(pState->filter.dTapSpacing);
  2597. nTail= (nTaps-1)/2;
  2598. dDCa= dDC-1.0;
  2599. dDCb= dDC+1.0;
  2600. dDC *= Sum_d(pdTaps, nTaps);
  2601. dDCa *= Sum_d(pdTaps, nTaps);
  2602. dDCb *= Sum_d(pdTaps, nTaps);
  2603. for (i= 0; i< pState->SamplesPerLine; i++) {
  2604. index= float2long(i+nTail*nTapSpacing);
  2605. SASSERT(index >= 0 && index < sizeof(pState->pdScratch1)/sizeof(Double));
  2606. pdLine[index]= (Double) pbSamples[i];
  2607. }
  2608. for (i= 0; i< nTail*nTapSpacing; i++) {
  2609. index= i;
  2610. SASSERT(index >= 0 && index < sizeof(pState->pdScratch1)/sizeof(Double));
  2611. pdLine[index]= dDC;
  2612. index= float2long(i + pState->SamplesPerLine + nTail * nTapSpacing);
  2613. SASSERT(index >= 0 && index < sizeof(pState->pdScratch1)/sizeof(Double));
  2614. pdLine[index]= dDC;
  2615. }
  2616. nSamplePos += 20;
  2617. for (i= 0; i< NABTS_BYTES_PER_LINE; i++) {
  2618. int j;
  2619. int nByte= 0;
  2620. /*int nBytea= 0;*/
  2621. /*int nByteb= 0;*/
  2622. for (j= 0; j< 8; j++) {
  2623. Double dSum;
  2624. nByte >>= 1;
  2625. #if 0
  2626. nBytea >>= 1;
  2627. nByteb >>= 1;
  2628. #endif
  2629. SASSERT(nSamplePos-20 >= 0);
  2630. SASSERT(nSamplePos+20 < sizeof(pState->pdScratch1)/sizeof(Double));
  2631. dSum=
  2632. pdTaps[ 0] * pdLine[nSamplePos-20] +
  2633. pdTaps[ 1] * pdLine[nSamplePos-18] +
  2634. pdTaps[ 2] * pdLine[nSamplePos-16] +
  2635. pdTaps[ 3] * pdLine[nSamplePos-14] +
  2636. pdTaps[ 4] * pdLine[nSamplePos-12] +
  2637. pdTaps[ 5] * pdLine[nSamplePos-10] +
  2638. pdTaps[ 6] * pdLine[nSamplePos- 8] +
  2639. pdTaps[ 7] * pdLine[nSamplePos- 6] +
  2640. pdTaps[ 8] * pdLine[nSamplePos- 4] +
  2641. pdTaps[ 9] * pdLine[nSamplePos- 2] +
  2642. pdTaps[10] * pdLine[nSamplePos+ 0] +
  2643. pdTaps[11] * pdLine[nSamplePos+ 2] +
  2644. pdTaps[12] * pdLine[nSamplePos+ 4] +
  2645. pdTaps[13] * pdLine[nSamplePos+ 6] +
  2646. pdTaps[14] * pdLine[nSamplePos+ 8] +
  2647. pdTaps[15] * pdLine[nSamplePos+10] +
  2648. pdTaps[16] * pdLine[nSamplePos+12] +
  2649. pdTaps[17] * pdLine[nSamplePos+14] +
  2650. pdTaps[18] * pdLine[nSamplePos+16] +
  2651. pdTaps[19] * pdLine[nSamplePos+18] +
  2652. pdTaps[20] * pdLine[nSamplePos+20];
  2653. if (dSum > dDC) nByte |= 128;
  2654. #if 0
  2655. if (dSum > dDCa) nBytea |= 128;
  2656. if (dSum > dDCb) nByteb |= 128;
  2657. #endif
  2658. nSamplePos += 5;
  2659. }
  2660. pbBuf[i]= nByte;
  2661. #if 0
  2662. pbBufa[i]= nBytea;
  2663. pbBufb[i]= nByteb;
  2664. #endif
  2665. }
  2666. {
  2667. /*fec_error_class std, a, b;*/
  2668. unsigned char *pbWinner;
  2669. #if 0
  2670. if (nFECType == NDSP_BUNDLE_FEC_1) {
  2671. std= check_fec(pbBuf);
  2672. a= check_fec(pbBufa);
  2673. b= check_fec(pbBufb);
  2674. if (std <= a && std <= b) {
  2675. pbWinner= pbBuf;
  2676. }
  2677. else if (a<b) {
  2678. pbWinner= pbBufa;
  2679. } else {
  2680. pbWinner= pbBufb;
  2681. }
  2682. }
  2683. else
  2684. #endif
  2685. {
  2686. pbWinner= pbBuf;
  2687. }
  2688. for (i= 0; i< pState->SamplesPerLine; i++) {
  2689. pbDest[i]= pbWinner[i];
  2690. }
  2691. }
  2692. pdLine= NULL;
  2693. pState->bUsingScratch1= FALSE;
  2694. return 0;
  2695. }
  2696. /* Get bits for variable fixed taps.
  2697. Since in the general case this is inefficient, we check
  2698. for the various variable-tap filter configurations we use.
  2699. See NDSPGetBits for more information */
  2700. int NDSPGetBits_var(unsigned char *pbDest, unsigned char *pbSamples,
  2701. int nSyncStart, double dDC, NDSPState *pState,
  2702. int nFECType)
  2703. {
  2704. int nSamplePos= nSyncStart;
  2705. int i;
  2706. Double *pdLine;
  2707. Double *pdTaps;
  2708. int *pnTapLocs;
  2709. int nTaps;
  2710. int nFirstTap;
  2711. int nLastTap;
  2712. int index;
  2713. int iSampRate = float2long(pState->dSampleRate);
  2714. SASSERT(pState);
  2715. if (FilterIsVar(&pState->filter, 3, 11, 9, 0)) {
  2716. /* Taps are -33, -30 ... 24, 27 */
  2717. return NDSPGetBits_var_3_11_9_0(pbDest, pbSamples, nSyncStart, dDC,
  2718. pState, nFECType);
  2719. }
  2720. if (FilterIsVar(&pState->filter, 3, 11, 9, 2)) {
  2721. /* Taps are -33, -30 ... 24, 27, X, Y
  2722. where X and Y are any number */
  2723. return NDSPGetBits_var_3_11_9_2(pbDest, pbSamples, nSyncStart, dDC,
  2724. pState, nFECType);
  2725. }
  2726. if (FilterIsVar(&pState->filter, 2, 11, 9, 2)) {
  2727. return NDSPGetBits_var_2_11_9_2(pbDest, pbSamples, nSyncStart, dDC,
  2728. pState, nFECType);
  2729. }
  2730. if (FilterIsVar(&pState->filter, 5, 2, 2, 0)) {
  2731. /* Taps are -10, -5, 0, 5, 10 */
  2732. return NDSPGetBits_var_5_2_2_0(pbDest, pbSamples, nSyncStart, dDC,
  2733. pState, nFECType);
  2734. }
  2735. /* ===== add another "case" for speedy filter for 4x here ===== */
  2736. /* If we get this far in the field, we're probably doing the wrong
  2737. thing!
  2738. See NDSPGetBits for the general algorithm followed */
  2739. // TODO - BOTH 4x and 4.7x
  2740. #ifdef DEBUG
  2741. {
  2742. static int nWarning= 0;
  2743. nWarning++;
  2744. if ((nWarning % 1024) == 0) {
  2745. debug_printf(("Warning: doing slow convolution\n"));
  2746. }
  2747. }
  2748. #endif //DEBUG
  2749. nFirstTap= pState->filter.nMinTap;
  2750. nLastTap= pState->filter.nMaxTap;
  2751. SASSERT(!pState->bUsingScratch1);
  2752. pState->bUsingScratch1= __LINE__;
  2753. pdLine= pState->pdScratch1;
  2754. SASSERT(sizeof(pState->pdScratch1)/sizeof(Double) >=
  2755. pState->SamplesPerLine-nFirstTap+nLastTap);
  2756. pdTaps= pState->filter.pdTaps;
  2757. pnTapLocs= pState->filter.pnTapLocs;
  2758. nTaps= pState->filter.nTaps;
  2759. dDC *= Sum_d(pdTaps, nTaps);
  2760. for (i= 0; i< pState->SamplesPerLine; i++) {
  2761. index= i-nFirstTap;
  2762. SASSERT(index >= 0 && index < sizeof(pState->pdScratch1)/sizeof(Double));
  2763. pdLine[index]= (Double) pbSamples[i];
  2764. }
  2765. for (i= 0; i< -nFirstTap; i++) {
  2766. index= i;
  2767. SASSERT(index >= 0 && index < sizeof(pState->pdScratch1)/sizeof(Double));
  2768. pdLine[index]= dDC;
  2769. }
  2770. for (i= 0; i< nLastTap; i++) {
  2771. index= i + pState->SamplesPerLine -nFirstTap;
  2772. SASSERT(index >= 0 && index < sizeof(pState->pdScratch1)/sizeof(Double));
  2773. pdLine[index]= dDC;
  2774. }
  2775. nSamplePos += (-nFirstTap);
  2776. for (i= 0; i< NABTS_BYTES_PER_LINE; i++) {
  2777. int j,k;
  2778. int nByte= 0;
  2779. for (j= 0; j< 8; j++) {
  2780. Double dSum= 0.0;
  2781. nByte >>= 1;
  2782. SASSERT(nSamplePos+nFirstTap >= 0);
  2783. SASSERT(nSamplePos+nLastTap < sizeof(pState->pdScratch1)/sizeof(Double));
  2784. for (k= 0; k< nTaps; k++) {
  2785. dSum += pdTaps[k] * pdLine[nSamplePos + pnTapLocs[k]];
  2786. }
  2787. if (dSum > dDC) nByte |= 128;
  2788. nSamplePos += iSampRate;
  2789. }
  2790. pbDest[i]= nByte;
  2791. }
  2792. pdLine= NULL;
  2793. pState->bUsingScratch1= FALSE;
  2794. return 0;
  2795. }
  2796. /* Check to see if the variable-tap filter matches the configuration
  2797. we're looking for.
  2798. nSpacing is the spacing between the "main" taps.
  2799. nLeft is the number of "main" taps less than zero.
  2800. nRight is the number of "main" taps greater than zero.
  2801. nExtra is the number of "extra" variable taps at the end which don't
  2802. necessarily fit in with nSpacing spacing.
  2803. We want to match taps looking like:
  2804. -nLeft * nSpacing, -(nLeft-1) * nSpacing, ... 0, ... (nRight-1) * nSpacing,
  2805. nRight * nSpacing, X-sub-1, X-sub-2, ... X-sub-nExtra
  2806. Where X-sub-N can be any number */
  2807. BOOL FilterIsVar(FIRFilter *pFilter, int nSpacing, int nLeft, int nRight,
  2808. int nExtra)
  2809. {
  2810. int i;
  2811. if (!pFilter->bVariableTaps) return FALSE;
  2812. if (nLeft + nRight + 1 + nExtra != pFilter->nTaps) return FALSE;
  2813. for (i= 0; i< nLeft + nRight + 1; i++) {
  2814. if (pFilter->pnTapLocs[i] != (i-nLeft)*nSpacing) return FALSE;
  2815. }
  2816. return TRUE;
  2817. }
  2818. /* Special case DSP for variable-tap filter having taps
  2819. -10, -5, 0, 5, 10.
  2820. See NDSPGetBits for information on the algorithm, and
  2821. NDSPGetBits_var for more details */
  2822. int NDSPGetBits_var_5_2_2_0(unsigned char *pbDest, unsigned char *pbSamples,
  2823. int nSyncStart, double dDC, NDSPState *pState,
  2824. int nFECType)
  2825. {
  2826. int nSamplePos= nSyncStart;
  2827. int i;
  2828. Double *pdLine;
  2829. Double *pdTaps;
  2830. int *pnTapLocs;
  2831. int nTaps;
  2832. int nFirstTap;
  2833. int nLastTap;
  2834. int index;
  2835. int nSampleInc = float2long(pState->dSampleRate);
  2836. SASSERT(pState);
  2837. SASSERT(FilterIsVar(&pState->filter,5,2,2,0));
  2838. nFirstTap= pState->filter.nMinTap;
  2839. nLastTap= pState->filter.nMaxTap;
  2840. SASSERT(!pState->bUsingScratch1);
  2841. pState->bUsingScratch1= __LINE__;
  2842. pdLine= pState->pdScratch1;
  2843. SASSERT(sizeof(pState->pdScratch1)/sizeof(Double) >=
  2844. pState->SamplesPerLine-nFirstTap+nLastTap);
  2845. pdTaps= pState->filter.pdTaps;
  2846. pnTapLocs= pState->filter.pnTapLocs;
  2847. nTaps= pState->filter.nTaps;
  2848. dDC *= Sum_d(pdTaps, nTaps);
  2849. for (i= 0; i< pState->SamplesPerLine; i++) {
  2850. index= i-nFirstTap;
  2851. SASSERT(index >= 0 && index < sizeof(pState->pdScratch1)/sizeof(Double));
  2852. pdLine[index]= (Double) pbSamples[i];
  2853. }
  2854. for (i= 0; i< -nFirstTap; i++) {
  2855. index= i;
  2856. SASSERT(index >= 0 && index < sizeof(pState->pdScratch1)/sizeof(Double));
  2857. pdLine[index]= dDC;
  2858. }
  2859. for (i= 0; i< nLastTap; i++) {
  2860. index= i + pState->SamplesPerLine -nFirstTap;
  2861. SASSERT(index >= 0 && index < sizeof(pState->pdScratch1)/sizeof(Double));
  2862. pdLine[index]= dDC;
  2863. }
  2864. nSamplePos += (-nFirstTap);
  2865. for (i= 0; i< NABTS_BYTES_PER_LINE; i++) {
  2866. int j;
  2867. int nByte= 0;
  2868. for (j= 0; j< 8; j++) {
  2869. Double dSum= 0.0;
  2870. nByte >>= 1;
  2871. SASSERT(nSamplePos+nFirstTap >= 0);
  2872. SASSERT(nSamplePos+nLastTap < sizeof(pState->pdScratch1)/sizeof(Double));
  2873. dSum=
  2874. pdTaps[0] * pdLine[nSamplePos - 10] +
  2875. pdTaps[1] * pdLine[nSamplePos - 5] +
  2876. pdTaps[2] * pdLine[nSamplePos + 0] +
  2877. pdTaps[3] * pdLine[nSamplePos + 5] +
  2878. pdTaps[4] * pdLine[nSamplePos + 10];
  2879. if (dSum > dDC) nByte |= 128;
  2880. nSamplePos += nSampleInc;
  2881. }
  2882. pbDest[i]= nByte;
  2883. }
  2884. pdLine= NULL;
  2885. pState->bUsingScratch1= FALSE;
  2886. return 0;
  2887. }
  2888. /* Special case DSP for variable-tap filter having taps
  2889. -33, -30 ... 24, 27
  2890. See NDSPGetBits for information on the algorithm, and
  2891. NDSPGetBits_var for more details */
  2892. int NDSPGetBits_var_3_11_9_0(unsigned char *pbDest, unsigned char *pbSamples,
  2893. int nSyncStart, double dDC, NDSPState *pState,
  2894. int nFECType)
  2895. {
  2896. int nSamplePos= nSyncStart;
  2897. int i;
  2898. Double *pdLine;
  2899. Double *pdTaps;
  2900. int *pnTapLocs;
  2901. int nTaps;
  2902. int nFirstTap;
  2903. int nLastTap;
  2904. int index;
  2905. int nSampleInc = float2long(pState->dSampleRate);
  2906. SASSERT(pState);
  2907. SASSERT(FilterIsVar(&pState->filter,3,11,9,0));
  2908. nFirstTap= pState->filter.nMinTap;
  2909. nLastTap= pState->filter.nMaxTap;
  2910. SASSERT(!pState->bUsingScratch1);
  2911. pState->bUsingScratch1= __LINE__;
  2912. pdLine= pState->pdScratch1;
  2913. SASSERT(sizeof(pState->pdScratch1)/sizeof(Double) >=
  2914. pState->SamplesPerLine-nFirstTap+nLastTap);
  2915. pdTaps= pState->filter.pdTaps;
  2916. pnTapLocs= pState->filter.pnTapLocs;
  2917. nTaps= pState->filter.nTaps;
  2918. dDC *= Sum_d(pdTaps, nTaps);
  2919. for (i= 0; i< pState->SamplesPerLine; i++) {
  2920. index= i-nFirstTap;
  2921. SASSERT(index >= 0 && index < sizeof(pState->pdScratch1)/sizeof(Double));
  2922. pdLine[index]= (Double) pbSamples[i];
  2923. }
  2924. for (i= 0; i< -nFirstTap; i++) {
  2925. index= i;
  2926. SASSERT(index >= 0 && index < sizeof(pState->pdScratch1)/sizeof(Double));
  2927. pdLine[index]= dDC;
  2928. }
  2929. for (i= 0; i< nLastTap; i++) {
  2930. index= i + pState->SamplesPerLine -nFirstTap;
  2931. SASSERT(index >= 0 && index < sizeof(pState->pdScratch1)/sizeof(Double));
  2932. pdLine[index]= dDC;
  2933. }
  2934. nSamplePos += (-nFirstTap);
  2935. for (i= 0; i< NABTS_BYTES_PER_LINE; i++) {
  2936. int j;
  2937. int nByte= 0;
  2938. for (j= 0; j< 8; j++) {
  2939. Double dSum= 0.0;
  2940. nByte >>= 1;
  2941. SASSERT(nSamplePos+nFirstTap >= 0);
  2942. SASSERT(nSamplePos+nLastTap < sizeof(pState->pdScratch1)/sizeof(Double));
  2943. dSum=
  2944. pdTaps[0] * pdLine[nSamplePos - 33] +
  2945. pdTaps[1] * pdLine[nSamplePos - 30] +
  2946. pdTaps[2] * pdLine[nSamplePos - 27] +
  2947. pdTaps[3] * pdLine[nSamplePos - 24] +
  2948. pdTaps[4] * pdLine[nSamplePos - 21] +
  2949. pdTaps[5] * pdLine[nSamplePos - 18] +
  2950. pdTaps[6] * pdLine[nSamplePos - 15] +
  2951. pdTaps[7] * pdLine[nSamplePos - 12] +
  2952. pdTaps[8] * pdLine[nSamplePos - 9] +
  2953. pdTaps[9] * pdLine[nSamplePos - 6] +
  2954. pdTaps[10] * pdLine[nSamplePos - 3] +
  2955. pdTaps[11] * pdLine[nSamplePos + 0] +
  2956. pdTaps[12] * pdLine[nSamplePos + 3] +
  2957. pdTaps[13] * pdLine[nSamplePos + 6] +
  2958. pdTaps[14] * pdLine[nSamplePos + 9] +
  2959. pdTaps[15] * pdLine[nSamplePos + 12] +
  2960. pdTaps[16] * pdLine[nSamplePos + 15] +
  2961. pdTaps[17] * pdLine[nSamplePos + 18] +
  2962. pdTaps[18] * pdLine[nSamplePos + 21] +
  2963. pdTaps[19] * pdLine[nSamplePos + 24] +
  2964. pdTaps[20] * pdLine[nSamplePos + 27];
  2965. if (dSum > dDC) nByte |= 128;
  2966. nSamplePos += nSampleInc;
  2967. }
  2968. pbDest[i]= nByte;
  2969. }
  2970. pdLine= NULL;
  2971. pState->bUsingScratch1= FALSE;
  2972. return 0;
  2973. }
  2974. /* Special case DSP for variable-tap filter having taps
  2975. -33, -30 ... 24, 27, X, Y
  2976. where X and Y are any number
  2977. See NDSPGetBits for information on the algorithm, and
  2978. NDSPGetBits_var for more details */
  2979. int NDSPGetBits_var_3_11_9_2(unsigned char *pbDest, unsigned char *pbSamples,
  2980. int nSyncStart, double dDC, NDSPState *pState,
  2981. int nFECType)
  2982. {
  2983. int nSamplePos= nSyncStart;
  2984. int i;
  2985. Double *pdLine;
  2986. Double *pdTaps;
  2987. int *pnTapLocs;
  2988. int nTaps;
  2989. int nFirstTap;
  2990. int nLastTap;
  2991. int index;
  2992. int nSampleInc = float2long(pState->dSampleRate);
  2993. SASSERT(pState);
  2994. SASSERT(FilterIsVar(&pState->filter,3,11,9,2));
  2995. nFirstTap= pState->filter.nMinTap;
  2996. nLastTap= pState->filter.nMaxTap;
  2997. SASSERT(!pState->bUsingScratch1);
  2998. pState->bUsingScratch1= __LINE__;
  2999. pdLine= pState->pdScratch1;
  3000. SASSERT(sizeof(pState->pdScratch1)/sizeof(Double) >=
  3001. pState->SamplesPerLine-nFirstTap+nLastTap);
  3002. pdTaps= pState->filter.pdTaps;
  3003. pnTapLocs= pState->filter.pnTapLocs;
  3004. nTaps= pState->filter.nTaps;
  3005. dDC *= Sum_d(pdTaps, nTaps);
  3006. for (i= 0; i< pState->SamplesPerLine; i++) {
  3007. index= i-nFirstTap;
  3008. SASSERT(index >= 0 && index < sizeof(pState->pdScratch1)/sizeof(Double));
  3009. pdLine[index]= (Double) pbSamples[i];
  3010. }
  3011. for (i= 0; i< -nFirstTap; i++) {
  3012. index= i;
  3013. SASSERT(index >= 0 && index < sizeof(pState->pdScratch1)/sizeof(Double));
  3014. pdLine[index]= dDC;
  3015. }
  3016. for (i= 0; i< nLastTap; i++) {
  3017. index= i + pState->SamplesPerLine -nFirstTap;
  3018. SASSERT(index >= 0 && index < sizeof(pState->pdScratch1)/sizeof(Double));
  3019. pdLine[index]= dDC;
  3020. }
  3021. nSamplePos += (-nFirstTap);
  3022. for (i= 0; i< NABTS_BYTES_PER_LINE; i++) {
  3023. int j;
  3024. int nByte= 0;
  3025. for (j= 0; j< 8; j++) {
  3026. Double dSum= 0.0;
  3027. nByte >>= 1;
  3028. SASSERT(nSamplePos+nFirstTap >= 0);
  3029. SASSERT(nSamplePos+nLastTap < sizeof(pState->pdScratch1)/sizeof(Double));
  3030. dSum=
  3031. pdTaps[0] * pdLine[nSamplePos - 33] +
  3032. pdTaps[1] * pdLine[nSamplePos - 30] +
  3033. pdTaps[2] * pdLine[nSamplePos - 27] +
  3034. pdTaps[3] * pdLine[nSamplePos - 24] +
  3035. pdTaps[4] * pdLine[nSamplePos - 21] +
  3036. pdTaps[5] * pdLine[nSamplePos - 18] +
  3037. pdTaps[6] * pdLine[nSamplePos - 15] +
  3038. pdTaps[7] * pdLine[nSamplePos - 12] +
  3039. pdTaps[8] * pdLine[nSamplePos - 9] +
  3040. pdTaps[9] * pdLine[nSamplePos - 6] +
  3041. pdTaps[10] * pdLine[nSamplePos - 3] +
  3042. pdTaps[11] * pdLine[nSamplePos + 0] +
  3043. pdTaps[12] * pdLine[nSamplePos + 3] +
  3044. pdTaps[13] * pdLine[nSamplePos + 6] +
  3045. pdTaps[14] * pdLine[nSamplePos + 9] +
  3046. pdTaps[15] * pdLine[nSamplePos + 12] +
  3047. pdTaps[16] * pdLine[nSamplePos + 15] +
  3048. pdTaps[17] * pdLine[nSamplePos + 18] +
  3049. pdTaps[18] * pdLine[nSamplePos + 21] +
  3050. pdTaps[19] * pdLine[nSamplePos + 24] +
  3051. pdTaps[20] * pdLine[nSamplePos + 27] +
  3052. pdTaps[21] * pdLine[nSamplePos + pnTapLocs[21]] +
  3053. pdTaps[22] * pdLine[nSamplePos + pnTapLocs[22]];
  3054. if (dSum > dDC) nByte |= 128;
  3055. nSamplePos += nSampleInc;
  3056. }
  3057. pbDest[i]= nByte;
  3058. }
  3059. pdLine= NULL;
  3060. pState->bUsingScratch1= FALSE;
  3061. return 0;
  3062. }
  3063. int NDSPGetBits_var_2_11_9_2(unsigned char *pbDest, unsigned char *pbSamples,
  3064. int nSyncStart, double dDC, NDSPState *pState,
  3065. int nFECType)
  3066. {
  3067. int nSamplePos= nSyncStart;
  3068. int i;
  3069. Double *pdLine;
  3070. Double *pdTaps;
  3071. int *pnTapLocs;
  3072. int nTaps;
  3073. int nFirstTap;
  3074. int nLastTap;
  3075. int index;
  3076. int nSampleInc = float2long(pState->dSampleRate);
  3077. SASSERT(pState);
  3078. SASSERT(FilterIsVar(&pState->filter,2,11,9,2));
  3079. nFirstTap= pState->filter.nMinTap;
  3080. nLastTap= pState->filter.nMaxTap;
  3081. SASSERT(!pState->bUsingScratch1);
  3082. pState->bUsingScratch1= __LINE__;
  3083. pdLine= pState->pdScratch1;
  3084. SASSERT(sizeof(pState->pdScratch1)/sizeof(Double) >=
  3085. pState->SamplesPerLine-nFirstTap+nLastTap);
  3086. pdTaps= pState->filter.pdTaps;
  3087. pnTapLocs= pState->filter.pnTapLocs;
  3088. nTaps= pState->filter.nTaps;
  3089. dDC *= Sum_d(pdTaps, nTaps);
  3090. for (i= 0; i< pState->SamplesPerLine; i++) {
  3091. index= i-nFirstTap;
  3092. SASSERT(index >= 0 && index < sizeof(pState->pdScratch1)/sizeof(Double));
  3093. pdLine[index]= (Double) pbSamples[i];
  3094. }
  3095. for (i= 0; i< -nFirstTap; i++) {
  3096. index= i;
  3097. SASSERT(index >= 0 && index < sizeof(pState->pdScratch1)/sizeof(Double));
  3098. pdLine[index]= dDC;
  3099. }
  3100. for (i= 0; i< nLastTap; i++) {
  3101. index= i + pState->SamplesPerLine -nFirstTap;
  3102. SASSERT(index >= 0 && index < sizeof(pState->pdScratch1)/sizeof(Double));
  3103. pdLine[index]= dDC;
  3104. }
  3105. nSamplePos += (-nFirstTap);
  3106. for (i= 0; i< NABTS_BYTES_PER_LINE; i++) {
  3107. int j;
  3108. int nByte= 0;
  3109. for (j= 0; j< 8; j++) {
  3110. Double dSum= 0.0;
  3111. nByte >>= 1;
  3112. SASSERT(nSamplePos+nFirstTap >= 0);
  3113. SASSERT(nSamplePos+nLastTap < sizeof(pState->pdScratch1)/sizeof(Double));
  3114. dSum=
  3115. pdTaps[0] * pdLine[nSamplePos - 22] +
  3116. pdTaps[1] * pdLine[nSamplePos - 20] +
  3117. pdTaps[2] * pdLine[nSamplePos - 18] +
  3118. pdTaps[3] * pdLine[nSamplePos - 16] +
  3119. pdTaps[4] * pdLine[nSamplePos - 14] +
  3120. pdTaps[5] * pdLine[nSamplePos - 12] +
  3121. pdTaps[6] * pdLine[nSamplePos - 10] +
  3122. pdTaps[7] * pdLine[nSamplePos - 8] +
  3123. pdTaps[8] * pdLine[nSamplePos - 6] +
  3124. pdTaps[9] * pdLine[nSamplePos - 4] +
  3125. pdTaps[10] * pdLine[nSamplePos - 2] +
  3126. pdTaps[11] * pdLine[nSamplePos + 0] +
  3127. pdTaps[12] * pdLine[nSamplePos + 2] +
  3128. pdTaps[13] * pdLine[nSamplePos + 4] +
  3129. pdTaps[14] * pdLine[nSamplePos + 6] +
  3130. pdTaps[15] * pdLine[nSamplePos + 8] +
  3131. pdTaps[16] * pdLine[nSamplePos + 10] +
  3132. pdTaps[17] * pdLine[nSamplePos + 12] +
  3133. pdTaps[18] * pdLine[nSamplePos + 14] +
  3134. pdTaps[19] * pdLine[nSamplePos + 16] +
  3135. pdTaps[20] * pdLine[nSamplePos + 18] +
  3136. pdTaps[21] * pdLine[nSamplePos + pnTapLocs[21]] +
  3137. pdTaps[22] * pdLine[nSamplePos + pnTapLocs[22]];
  3138. if (dSum > dDC) nByte |= 128;
  3139. nSamplePos += nSampleInc;
  3140. }
  3141. pbDest[i]= nByte;
  3142. }
  3143. pdLine= NULL;
  3144. pState->bUsingScratch1= FALSE;
  3145. return 0;
  3146. }
  3147. #ifdef DEBUG
  3148. int debug_equalize= 0;
  3149. void print_filter(FIRFilter *pFilt)
  3150. {
  3151. if (pFilt->bVariableTaps) {
  3152. int i;
  3153. debug_printf(("[FIR %dV [", pFilt->nTaps));
  3154. for (i= 0; i< pFilt->nTaps; i++) {
  3155. if (i) debug_printf((" "));
  3156. debug_printf(("%d:%s", pFilt->pnTapLocs[i], flPrintf(pFilt->pdTaps[i],4)));
  3157. }
  3158. debug_printf(("]]"));
  3159. } else {
  3160. debug_printf(("[FIR %dx%s ", pFilt->nTaps, flPrintf(pFilt->dTapSpacing,3)));
  3161. printarray_d1(pFilt->pdTaps, pFilt->nTaps);
  3162. debug_printf(("]"));
  3163. }
  3164. }
  3165. void printarray_d1(Double *arr, int n)
  3166. {
  3167. int i;
  3168. debug_printf(("["));
  3169. for (i= 0; i< n; i++) {
  3170. if (i) debug_printf((" "));
  3171. debug_printf(("%s", flPrintf(arr[i], 2)));
  3172. }
  3173. debug_printf(("]"));
  3174. }
  3175. void printarray_ed1(Double *arr, int n)
  3176. {
  3177. int i;
  3178. debug_printf(("["));
  3179. for (i= 0; i< n; i++) {
  3180. if (i) debug_printf((" "));
  3181. debug_printf(("%s", flPrintf(arr[i], 4)));
  3182. }
  3183. debug_printf(("]"));
  3184. }
  3185. void printarray_d2(Double *arr, int a_size, int b_size)
  3186. {
  3187. int a,b;
  3188. for (a= 0; a< a_size; a++) {
  3189. debug_printf(("|"));
  3190. for (b= 0; b< b_size; b++) {
  3191. if (b) debug_printf((" "));
  3192. debug_printf(("%s", flPrintf(arr[a*b_size+b], 4)));
  3193. }
  3194. debug_printf(("|\n"));
  3195. }
  3196. debug_printf(("\n"));
  3197. }
  3198. void printarray_ed2(Double *arr, int a_size, int b_size)
  3199. {
  3200. int a,b;
  3201. for (a= 0; a< a_size; a++) {
  3202. debug_printf(("|"));
  3203. for (b= 0; b< b_size; b++) {
  3204. if (b) debug_printf((" "));
  3205. debug_printf(("%s", flPrintf(arr[a*b_size+b], 4)));
  3206. }
  3207. debug_printf(("|\n"));
  3208. }
  3209. debug_printf(("\n"));
  3210. }
  3211. #endif //DEBUG
  3212. #ifdef DEBUG_AREF
  3213. #define a2d(arr,a,b,abound,bbound) (EASSERT(0<=(a)&&(a)<(abound)), \
  3214. EASSERT(0<=(b)&&(b)<(bbound)), \
  3215. (arr)[(a)*(bbound)+(b)])
  3216. #else
  3217. #define a2d(arr,a,b,abound,bbound) ((arr)[(a)*(bbound)+(b)])
  3218. #endif
  3219. #ifdef DEBUG_AREF
  3220. #define a1d(arr,a,abound) (EASSERT(0<=(a)&&(a)<(abound)), \
  3221. (arr)[a])
  3222. #else
  3223. #define a1d(arr,a,abound) ((arr)[a])
  3224. #endif
  3225. #define aI(a,b) a2d(I,a,b,nFilterLength,nDesiredOutputLength)
  3226. #define aItI(a,b) a2d(ItI,a,b,nFilterLength,nFilterLength)
  3227. #define aIto(a) a1d(Ito,a,nFilterLength)
  3228. #define ao(a) a1d(o,a,nDesiredOutputLength)
  3229. #ifdef DEBUG_AREF
  3230. #define aInput(a) (EASSERT(nMinInputIndex<=(a)&&(a)<=nMaxInputIndex), \
  3231. pfInput[a])
  3232. #else
  3233. #define aInput(a) (pfInput[a])
  3234. #endif
  3235. #define EQ_TOL .000001
  3236. void SubtractMean_d(Double *pfOutput, Double *pfInput, int nLen)
  3237. {
  3238. int i;
  3239. Double dMean;
  3240. if (!nLen) return;
  3241. dMean= Mean_d(pfInput, nLen);
  3242. for (i= 0; i< nLen; i++) pfOutput[i]= pfInput[i] - dMean;
  3243. }
  3244. /* Modification of above:
  3245. Filter now has a variable number of taps */
  3246. void FillGCRSignals()
  3247. {
  3248. int i;
  3249. SubtractMean_d(g_pdGCRSignal1, g_pdGCRSignal1, g_nNabtsGcrSize);
  3250. for (i= 0; i< g_nNabtsGcrSize; i++) g_pdGCRSignal1[i] *= 100;
  3251. for (i= 0; i< g_nNabtsGcrSize; i++) g_pdGCRSignal2[i]= -g_pdGCRSignal1[i];
  3252. }
  3253. void NormalizeSyncSignal()
  3254. {
  3255. SubtractMean_d(g_pdSync, g_pdSync, g_nNabtsSyncSize);
  3256. }
  3257. Double Mean_d(Double *a, int a_size)
  3258. {
  3259. Double sum= 0.0;
  3260. int i;
  3261. SASSERT(a_size>0);
  3262. for (i= 0; i< a_size; i++) sum += a[i];
  3263. return sum / a_size;
  3264. }
  3265. Double Mean_8(unsigned char *a, int a_size)
  3266. {
  3267. int sum= 0.0;
  3268. int i;
  3269. SASSERT(a_size>0);
  3270. for (i= 0; i< a_size; i++) sum += a[i];
  3271. return ((Double) sum) / a_size;
  3272. }
  3273. void Mult_d(Double *dest, Double *src, int size, Double factor)
  3274. {
  3275. int i;
  3276. for (i= 0; i< size; i++) dest[i]= src[i]*factor;
  3277. }
  3278. int Sum_16(short *a, int a_size)
  3279. {
  3280. int ret= 0;
  3281. int i;
  3282. for (i= 0; i< a_size; i++) ret += a[i];
  3283. return ret;
  3284. }
  3285. /* Try to match the input signal with a given EqualizeMatch desired
  3286. signal template.
  3287. If there is an apparent match, return offset at which match is found. */
  3288. int MatchWithEqualizeSignal(NDSPState *pState,
  3289. unsigned char *pbSamples,
  3290. EqualizeMatch *eqm,
  3291. int *pnOffset,
  3292. Double *pdMaxval,
  3293. BOOL bNegativeOK)
  3294. {
  3295. Double dMaxval= 0.0;
  3296. Double dMinval= 0.0;
  3297. Double *pdSamples;
  3298. Double *pdConv;
  3299. int nMaxindex= 0;
  3300. int nMinindex= 0;
  3301. int i;
  3302. int nStatus= 0;
  3303. if (!pState || pState->uMagic != NDSP_STATE_MAGIC)
  3304. return NDSP_ERROR_ILLEGAL_NDSP_STATE;
  3305. SASSERT(!pState->bUsingScratch1);
  3306. SASSERT(sizeof(pState->pdScratch1)/sizeof(Double) >= pState->SamplesPerLine);
  3307. pdSamples= pState->pdScratch1;
  3308. pState->bUsingScratch1= __LINE__;
  3309. SASSERT(!pState->bUsingScratch2);
  3310. SASSERT(sizeof(pState->pdScratch2)/sizeof(Double) >= pState->SamplesPerLine);
  3311. pdConv= pState->pdScratch2;
  3312. pState->bUsingScratch2= __LINE__;
  3313. Copy_d_8(pdSamples, pbSamples, pState->SamplesPerLine);
  3314. Convolve_d_d_d(pdConv,
  3315. eqm->nSignalStartConv, eqm->nSignalEndConv,
  3316. pdSamples, 0, pState->SamplesPerLine-1, eqm->nSignalSampleRate,
  3317. eqm->pdSignal, 0, eqm->nSignalSize, 1,
  3318. TRUE);
  3319. for (i= eqm->nSignalStartConv; i<= eqm->nSignalEndConv; i++) {
  3320. if (pdConv[i] > dMaxval) {
  3321. dMaxval= pdConv[i];
  3322. nMaxindex= i;
  3323. }
  3324. if (pdConv[i] < dMinval) {
  3325. dMinval= pdConv[i];
  3326. nMinindex= i;
  3327. }
  3328. }
  3329. if ((-dMinval) > dMaxval && bNegativeOK) {
  3330. dMaxval= dMinval;
  3331. nMaxindex= nMinindex;
  3332. }
  3333. pdConv= NULL;
  3334. pState->bUsingScratch2= FALSE;
  3335. if (pnOffset) { (*pnOffset)= nMaxindex; }
  3336. if (pdMaxval) { (*pdMaxval)= dMaxval; }
  3337. pdSamples= NULL;
  3338. pState->bUsingScratch1= FALSE;
  3339. return nStatus;
  3340. }
  3341. /* Adaptive equalizer.
  3342. Set the coefficients on a simple FIR filter such that the input
  3343. waveform matches most closely the output (minimizing the sum of
  3344. the squares of the error terms).
  3345. i= input signal (vector)
  3346. o= desirect output signal (vector)
  3347. c= coefficients of FIR (vector)
  3348. e= error (scalar)
  3349. e= |convolution(i,c) - o|^2
  3350. We construct a matrix, I, such that
  3351. I*c = convolution(i,c)
  3352. Now we attempt to minimize e:
  3353. e= |I*c - o|^2
  3354. This can be solved as the simple system of linear equations:
  3355. (transpose(I)*I)*c = (transpose(I)*o)
  3356. */
  3357. /* pfDesiredOutput must be defined over 0 ... nDesiredOutputLength-1 */
  3358. /* pfInput must be defined over
  3359. -(nFilterLength-1)/2 ... nDesiredOutputLength-1 + (nFilterLength-1)/2
  3360. */
  3361. BOOL EqualizeVar(NDSPState *pState,
  3362. Double *pfInput, int nMinInputIndex, int nMaxInputIndex,
  3363. Double *pfDesiredOutput, int nDesiredOutputLength,
  3364. int nOutputSpacing, FIRFilter *pFilter)
  3365. {
  3366. int nFilterLength= pFilter->nTaps;
  3367. /* We spend most of our time in the n^3 multiplication of ItI,
  3368. so we choose the following order of indices */
  3369. /* ItI[nFilterLength][nFilterLength] */
  3370. /* (transpose(I) * I) */
  3371. Double *ItI;
  3372. /* Ito[nFilterLength] */
  3373. /* (transpose(I) * o) */
  3374. Double *Ito;
  3375. /* o[nDesiredOutputLength] */
  3376. Double *o;
  3377. BOOL bRet=FALSE;
  3378. int x,y,i;
  3379. SASSERT(!pState->bUsingScratch3);
  3380. SASSERT(sizeof(pState->pdScratch3) >=
  3381. nFilterLength * nFilterLength * sizeof(Double));
  3382. ItI= pState->pdScratch3;
  3383. pState->bUsingScratch3= __LINE__;
  3384. SASSERT(!pState->bUsingScratch4);
  3385. SASSERT(sizeof(pState->pdScratch4) >=
  3386. nFilterLength * sizeof(Double));
  3387. Ito= pState->pdScratch4;
  3388. pState->bUsingScratch4= __LINE__;
  3389. SASSERT(!pState->bUsingScratch5);
  3390. SASSERT(sizeof(pState->pdScratch5) >=
  3391. nDesiredOutputLength * sizeof(Double));
  3392. o= pState->pdScratch5;
  3393. pState->bUsingScratch5= __LINE__;
  3394. for (i= 0; i< nDesiredOutputLength; i++) {
  3395. o[i]= pfDesiredOutput[i];
  3396. }
  3397. /*memcpy(o, pfDesiredOutput, nDesiredOutputLength * sizeof(Double));*/
  3398. SASSERT(nFilterLength >= 1);
  3399. /* Create (It)I */
  3400. /* Since ItI is symmetric, we only have to do a little over
  3401. half the multiplies */
  3402. for (x= 0; x< nFilterLength; x++) {
  3403. for (y= 0; y<=x; y++) {
  3404. Double fSum= 0.0;
  3405. for (i= 0; i< nDesiredOutputLength; i++) {
  3406. fSum +=
  3407. aInput(pFilter->pnTapLocs[x]+nOutputSpacing*i) *
  3408. aInput(pFilter->pnTapLocs[y]+nOutputSpacing*i);
  3409. }
  3410. aItI(x,y)= aItI(y,x)= fSum;
  3411. }
  3412. }
  3413. /* Create (It)o */
  3414. for (x= 0; x< nFilterLength; x++) {
  3415. Double fSum= 0.0;
  3416. for (i= 0; i< nDesiredOutputLength; i++) {
  3417. fSum += aInput(pFilter->pnTapLocs[x]+nOutputSpacing*i) *
  3418. ao(i);
  3419. }
  3420. aIto(x)= fSum;
  3421. }
  3422. #ifdef DEBUG_VERBOSE
  3423. if (debug_equalize) {
  3424. debug_printf(("ItI:\n"));
  3425. printarray_ed2(ItI, nFilterLength, nFilterLength);
  3426. debug_printf(("Ito:\n"));
  3427. printarray_ed1(Ito, nFilterLength);
  3428. debug_printf(("\n"));
  3429. }
  3430. #endif
  3431. /* Solve (ItI)c = Ito
  3432. Since Gaussian elimination is the simplest, we've implemented it first.
  3433. We can state a priori that no two coefficients are even closely
  3434. dependent in the solution, given that the input and the output
  3435. are related enough to pass the original confidence test.
  3436. Therefore, if we find ItI to be to be close to singular, we can
  3437. just reject it the sample, claiming that the GCR locater didn't do
  3438. it's job. This should in fact never happen.
  3439. */
  3440. /* We treat ItI as having indices [y][x] to speed the following. */
  3441. for (y= 0; y< nFilterLength; y++) {
  3442. Double fMax= 0.0;
  3443. int nMax= 0;
  3444. /* Find the largest magnitude value in this column to swap with */
  3445. for (i= y; i< nFilterLength; i++) {
  3446. Double fVal= aItI(i,y);
  3447. if (fVal < 0) fVal = -fVal;
  3448. if (fVal > fMax) { fMax= fVal; nMax= i; }
  3449. }
  3450. /* Swap the rows */
  3451. if (fMax < EQ_TOL) {
  3452. #ifdef DEBUG_VERBOSE
  3453. debug_printf(("Near-singular matrix in Equalize\n"));
  3454. debug_printf(("Sync correlator must be broken\n"));
  3455. #endif
  3456. bRet= FALSE;
  3457. goto exit;
  3458. }
  3459. if (nMax != y) {
  3460. Double fTmp;
  3461. for (x= y; x< nFilterLength; x++) {
  3462. fTmp= aItI(y,x);
  3463. aItI(y,x)= aItI(nMax,x);
  3464. aItI(nMax,x)= fTmp;
  3465. }
  3466. fTmp= aIto(y);
  3467. aIto(y)= aIto(nMax);
  3468. aIto(nMax)= fTmp;
  3469. }
  3470. for (i= y+1; i< nFilterLength; i++) {
  3471. Double fDependence= aItI(i,y) / aItI(y,y);
  3472. aItI(i,y)= 0.0;
  3473. for (x= y+1; x< nFilterLength; x++) {
  3474. aItI(i,x) -= aItI(y,x) * fDependence;
  3475. }
  3476. aIto(i) -= aIto(y) * fDependence;
  3477. }
  3478. }
  3479. /* Now ItI is upper diagonal */
  3480. for (y= nFilterLength - 1; y >= 0; y--) {
  3481. for (i= 0; i< y; i++) {
  3482. Double fDependence= aItI(i,y) / aItI(y,y);
  3483. aItI(i,y)= 0.0;
  3484. aIto(i) -= aIto(y) * fDependence;
  3485. }
  3486. }
  3487. /* Now divide Ito by the diagonals */
  3488. for (i= 0; i< nFilterLength; i++) {
  3489. aIto(i) /= aItI(i,i);
  3490. }
  3491. /* Ito now contains the answer */
  3492. /*memcpy(pfFilter, Ito, nFilterLength * sizeof(Double));*/
  3493. for (i= 0; i< nFilterLength; i++) {
  3494. pFilter->pdTaps[i]= Ito[i];
  3495. }
  3496. bRet= 1;
  3497. exit:
  3498. pState->bUsingScratch3= FALSE;
  3499. pState->bUsingScratch4= FALSE;
  3500. pState->bUsingScratch5= FALSE;
  3501. return bRet;
  3502. }
  3503. int get_sync_samples(unsigned long newHZ)
  3504. {
  3505. debug_printf(("get_sync_samples(%lu) entered\n", newHZ));
  3506. memset(g_pdSync, 0, MAX_NABTS_SAMPLES_PER_LINE);
  3507. switch (newHZ) {
  3508. case KS_VBISAMPLINGRATE_4X_NABTS:
  3509. memcpy(g_pdSync, g_pdSync4, NABSYNC_SIZE);
  3510. break;
  3511. case KS_VBISAMPLINGRATE_47X_NABTS:
  3512. memcpy(g_pdSync, g_pdSync47, NABSYNC_SIZE);
  3513. break;
  3514. case KS_VBISAMPLINGRATE_5X_NABTS:
  3515. memcpy(g_pdSync, g_pdSync5, NABSYNC_SIZE);
  3516. break;
  3517. default:
  3518. // Unknown sampling rate
  3519. debug_printf(("get_sync_samples: unknown sampling rate %lu\n", newHZ));
  3520. debug_breakpoint();
  3521. break;
  3522. }
  3523. return g_nNabtsSyncSize;
  3524. }
  3525. int get_gcr_samples(unsigned long newHZ)
  3526. {
  3527. int i;
  3528. debug_printf(("get_gcr_samples(%lu) entered\n", newHZ));
  3529. switch (newHZ) {
  3530. case KS_VBISAMPLINGRATE_4X_NABTS:
  3531. for (i=0; i<GCR_SIZE; i++) g_pdGCRSignal1[i] = g_pdGCRSignal1_4[i];
  3532. break;
  3533. case KS_VBISAMPLINGRATE_47X_NABTS:
  3534. for (i=0; i<GCR_SIZE; i++) g_pdGCRSignal1[i] = g_pdGCRSignal1_47[i];
  3535. break;
  3536. case KS_VBISAMPLINGRATE_5X_NABTS:
  3537. for (i=0; i<GCR_SIZE; i++) g_pdGCRSignal1[i] = g_pdGCRSignal1_5[i];
  3538. break;
  3539. default:
  3540. // Unknown sampling rate
  3541. debug_printf(("get_gcr_samples: unknown sampling rate %lu\n", newHZ));
  3542. debug_breakpoint();
  3543. break;
  3544. }
  3545. return g_nNabtsGcrSize;
  3546. }
  3547. void onResample(double sample_multiple, int syncLen, int gcrLen)
  3548. {
  3549. double ratio;
  3550. int nSize;
  3551. int i;
  3552. debug_printf(("onResample(%s, %d, %d) entered\n",
  3553. flPrintf(sample_multiple, 2), syncLen, gcrLen));
  3554. /* sync */
  3555. eqmatchNabtsSync.nSignalSize = syncLen;
  3556. eqmatchNabtsSync.nSignalStartConv =
  3557. float2long((double)NABSYNC_START_DETECT * sample_multiple/5.0);
  3558. eqmatchNabtsSync.nSignalEndConv =
  3559. float2long((double)NABSYNC_END_DETECT * sample_multiple/5.0);
  3560. eqmatchNabtsSync.pdSignal = g_pdSync;
  3561. /* gcr */
  3562. eqmatchGCR1.nSignalSize = gcrLen;
  3563. eqmatchGCR1.nSignalStartConv =
  3564. float2long((double)GCR_START_DETECT * sample_multiple/5.0);
  3565. eqmatchGCR1.nSignalEndConv =
  3566. float2long((double)GCR_END_DETECT * sample_multiple/5.0);
  3567. eqmatchGCR1.pdSignal = g_pdGCRSignal1;
  3568. eqmatchGCR2.nSignalSize = gcrLen;
  3569. eqmatchGCR2.nSignalStartConv =
  3570. float2long((double)GCR_START_DETECT * sample_multiple/5.0);
  3571. eqmatchGCR2.nSignalEndConv =
  3572. float2long((double)GCR_END_DETECT * sample_multiple/5.0);
  3573. eqmatchGCR2.pdSignal = g_pdGCRSignal2;
  3574. /* scale index arrays for NDSPDetectConfidence: */
  3575. ratio = sample_multiple/5.0;
  3576. nSize = sizeof(pnGCRPositiveIndices)/sizeof(int);
  3577. for (i=0; i<nSize; i++)
  3578. pnGCRPositiveIndices[i] = float2long((double)pnGCRPositiveIndices[i]*ratio);
  3579. nSize = sizeof(pnGCRNegativeIndices)/sizeof(int);
  3580. for (i=0; i<nSize; i++)
  3581. pnGCRNegativeIndices[i] = float2long((double)pnGCRNegativeIndices[i]*ratio);
  3582. }
  3583. void NDSPComputeNewSampleRate(unsigned long new_rate, unsigned long old_rate)
  3584. {
  3585. int gcrLen, syncLen;
  3586. debug_printf(("NDSPComputeNewSampleRate(%lu, %lu) entered\n", new_rate, old_rate));
  3587. gcrLen = get_gcr_samples(new_rate);
  3588. syncLen = get_sync_samples(new_rate);
  3589. onResample((double)new_rate / KS_VBIDATARATE_NABTS, syncLen, gcrLen);
  3590. }
  3591. void NDSPReset()
  3592. {
  3593. memcpy(pnGCRPositiveIndices, pnGCRPositiveIndices0, 26*sizeof(int));
  3594. memcpy(pnGCRNegativeIndices, pnGCRNegativeIndices0, 19*sizeof(int));
  3595. memcpy(g_pdGCRSignal1, g_pdGCRSignal1_5, GCR_SIZE*sizeof(Double));
  3596. memset(g_pdGCRSignal2, 0, GCR_SIZE*sizeof(Double));
  3597. memcpy(g_pdSync, g_pdSync5, NABSYNC_SIZE*sizeof(Double));
  3598. eqmatchNabtsSync.nSignalSize = NABSYNC_SIZE;
  3599. eqmatchNabtsSync.nSignalStartConv = NABSYNC_START_DETECT;
  3600. eqmatchNabtsSync.nSignalEndConv = NABSYNC_END_DETECT;
  3601. eqmatchNabtsSync.pdSignal = g_pdSync;
  3602. eqmatchGCR1.nSignalSize = GCR_SIZE;
  3603. eqmatchGCR1.nSignalStartConv = GCR_START_DETECT;
  3604. eqmatchGCR1.nSignalEndConv = GCR_END_DETECT;
  3605. eqmatchGCR1.pdSignal = g_pdGCRSignal1;
  3606. eqmatchGCR2.nSignalSize = GCR_SIZE;
  3607. eqmatchGCR2.nSignalStartConv = GCR_START_DETECT;
  3608. eqmatchGCR2.nSignalEndConv = GCR_END_DETECT;
  3609. eqmatchGCR2.pdSignal = g_pdGCRSignal2;
  3610. }