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

311 lines
8.2 KiB

  1. /* Copyright (c) 1997 Microsoft Corporation. All Rights Reserved. */
  2. /*
  3. foreach /x/samp/starsight/kctswod1-512-raw0000 : ccfile 4
  4. open /x/samp/starsight/kctswod1-512-raw0000
  5. 1: 39/2/0/0/40 old algorithm
  6. new:
  7. 2: 5/5/5/27/1091 avg zero crossings for level
  8. 3: 6/5/7/26/1069 avg peaks for level
  9. */
  10. #include <stdio.h>
  11. #include <stdlib.h>
  12. #include "host.h"
  13. #include "ccdecode.h"
  14. static inline int iabs(int x) { return x>0?x:-x; }
  15. static void cc_compute_new_samplingrate(CCState *pState, unsigned long newRate)
  16. {
  17. int i;
  18. MASSERT(pState);
  19. pState->lastFreq = newRate;
  20. pState->period = (newRate * CC_MULTIPLE) / KS_VBIDATARATE_CC;
  21. for (i = 1; i <= 13; ++i)
  22. pState->cc_sync_points[i] = (i * pState->period) / (2*CC_MULTIPLE);
  23. pState->cc_sync_points[14] = (17 * pState->period) / (2*CC_MULTIPLE);
  24. pState->cc_sync_points[15] = (19 * pState->period) / (2*CC_MULTIPLE);
  25. }
  26. /* The CC decoder previously did not use any persistent state. However,
  27. this version does. These calls are now REQUIRED.
  28. */
  29. /* Create a new CC state */
  30. CCState *CCStateNew(CCState *mem)
  31. {
  32. unsigned short no_free = 0;
  33. if (NULL == mem)
  34. mem = malloc(sizeof (CCState));
  35. else
  36. no_free = 1;
  37. if (NULL != mem) {
  38. mem->no_free = no_free;
  39. mem->magic = CC_STATE_MAGIC_10;
  40. cc_compute_new_samplingrate(mem, KS_VBISAMPLINGRATE_5X_NABTS);
  41. }
  42. return (mem);
  43. }
  44. /* Destroy the CC state. */
  45. void CCStateDestroy(CCState *state) {
  46. MASSERT(state);
  47. if (state->magic != 0) {
  48. state->magic = 0;
  49. if (!state->no_free)
  50. free(state);
  51. }
  52. }
  53. int cc_find_sync(CCState *pState, unsigned char *data, int max_sync_loc) {
  54. int i;
  55. int cur_conv = 0;
  56. int best_conv;
  57. int best_conv_loc;
  58. for (i = 0; i < 15; i++) {
  59. int sub_conv = 0;
  60. int j;
  61. for (j = pState->cc_sync_points[i]; j < pState->cc_sync_points[i+1]; j++) {
  62. sub_conv += data[j];
  63. }
  64. if (i & 1) {
  65. cur_conv -= sub_conv;
  66. } else {
  67. cur_conv += sub_conv;
  68. }
  69. }
  70. best_conv = cur_conv;
  71. best_conv_loc = 0;
  72. for (i = 1; i < max_sync_loc; i++) {
  73. int j;
  74. for (j = 0; j < 15; j++) {
  75. if (j & 1) {
  76. cur_conv += data[(i-1)+pState->cc_sync_points[j]];
  77. cur_conv -= data[(i-1)+pState->cc_sync_points[j+1]];
  78. } else {
  79. cur_conv -= data[(i-1)+pState->cc_sync_points[j]];
  80. cur_conv += data[(i-1)+pState->cc_sync_points[j+1]];
  81. }
  82. }
  83. if (cur_conv > best_conv) {
  84. best_conv = cur_conv;
  85. best_conv_loc = i;
  86. }
  87. }
  88. return best_conv_loc * CC_MULTIPLE;
  89. }
  90. /* Given a CC scanline, and the location of the sync, compute the
  91. DC offset of the signal. (We compute this by taking the average
  92. of the value at the 14 synchronization points found with the
  93. above routine; except we don't actually divide by 14 at the end.) */
  94. int cc_level(CCState *pState, unsigned char *data, int origin) {
  95. int i;
  96. int offset;
  97. int res = 0;
  98. for (i = 0, offset = origin; i < 7; i++, offset += pState->period) {
  99. res += CC_DATA(data, offset) + CC_DATA(data, offset + pState->period/2);
  100. }
  101. return res; /* Times 14! */
  102. }
  103. /* Given a CC scanline, the location of the sync, and the DC offset of
  104. the signal, check the "quality" of the signal (in particular, we
  105. want to determine whether this really is CC data). We do this by
  106. looking at 35 points in the signal and making sure that the values
  107. at those points agree with our expectations; we look at 26 points
  108. in the actual sync period (2 points in each "peak" and each "valley")
  109. and 9 points in the "check bits" (3 in each bit).
  110. This gives us a quality number between 0 and 35, where we would
  111. expect random noise to give us about 35/2. We map the range
  112. 0...35 to -1000...1000 before returning. */
  113. int cc_quality(CCState *pState, unsigned char *data, int origin, int level) {
  114. int i;
  115. int conf = 0;
  116. int offset;
  117. for (i = 0, offset = origin; i < 7; i++, offset += pState->period) {
  118. int ind_hi = (offset + pState->period/4)/CC_MULTIPLE;
  119. /* check 2 points in a "peak" */
  120. if (data[ind_hi-5] >= level) {
  121. conf++;
  122. }
  123. if (data[ind_hi+5] >= level) {
  124. conf++;
  125. }
  126. /* check 2 points in a "valley" */
  127. if (i < 6) {
  128. int ind_lo = (offset + 3*pState->period/4)/CC_MULTIPLE;
  129. if (data[ind_lo-5] < level) {
  130. conf++;
  131. }
  132. if (data[ind_lo+5] < level) {
  133. conf++;
  134. }
  135. }
  136. }
  137. for (i = 0, offset = origin + 7*pState->period; i < 3; i++, offset += pState->period) {
  138. /* check 3 points in a check bit */
  139. int ind = offset/CC_MULTIPLE;
  140. if (i < 2) {
  141. if (data[ind-10] < level) {
  142. conf++;
  143. }
  144. if (data[ind] < level) {
  145. conf++;
  146. }
  147. if (data[ind+10] < level) {
  148. conf++;
  149. }
  150. } else {
  151. if (data[ind-10] >= level) {
  152. conf++;
  153. }
  154. if (data[ind] >= level) {
  155. conf++;
  156. }
  157. if (data[ind+10] >= level) {
  158. conf++;
  159. }
  160. }
  161. }
  162. /* Now "conf" is a number between 0 and 35.
  163. If the input were random noise, we would expect "conf" to be about
  164. 35/2. We want to map 35/2 to 0 and 35 to 1000. (This actually maps
  165. 0 to -1000, 35/2 to 15, and 35 to 1030. Close enough.) */
  166. return (conf*58)-1000;
  167. }
  168. /* The main entry point for this file. Decodes a CC scanline into
  169. the two-byte "dest", and returns stats on the decoding. */
  170. int CCDecodeLine(unsigned char *dest, CCLineStats *stats,
  171. unsigned char *samples, CCState *state,
  172. KS_VBIINFOHEADER *pVBIINFO) {
  173. int origin;
  174. int quality;
  175. int level;
  176. int bits;
  177. int i;
  178. int offset;
  179. MASSERT(state);
  180. if (!state || !MCHECK(state))
  181. return CC_ERROR_ILLEGAL_STATE;
  182. if (stats->nSize != sizeof(*stats))
  183. return CC_ERROR_ILLEGAL_STATS;
  184. // Now check to see if we need to recompute for a different sampling rate
  185. if (state->lastFreq != pVBIINFO->SamplingFrequency)
  186. cc_compute_new_samplingrate(state, pVBIINFO->SamplingFrequency);
  187. #ifdef OLD_SYNC
  188. {
  189. int nOffsetData, nOffsetSamples;
  190. int offsets_err;
  191. /* Use the provided KS_VBIINFOHEADER to adjust the data so our
  192. various hardcoded constants are appropriate. */
  193. offsets_err = CCComputeOffsets(pVBIINFO, &nOffsetData, &nOffsetSamples);
  194. if (offsets_err > 0)
  195. return offsets_err;
  196. samples += nOffsetSamples;
  197. }
  198. #endif //OLD_SYNC
  199. origin = cc_find_sync(state, samples,
  200. pVBIINFO->SamplesPerLine
  201. - ((25*state->period)/CC_MULTIPLE) - 5);
  202. /* Find the DC offset of the signal (times 14) */
  203. level = cc_level(state, samples, origin + state->period/4); /* Times 14! */
  204. quality = cc_quality(state, samples, origin, level/14);
  205. /* Accumulate the actual data into "bits". */
  206. bits = 0;
  207. /* Start at the right and scan left; read 19 bits into "bits".
  208. (These are the 16 data bits and 3 check bits.) */
  209. for (i = 0, offset = origin + 25*state->period; i < 19; i++, offset -= state->period) {
  210. int ind= offset / CC_MULTIPLE;
  211. int measured_level;
  212. bits <<= 1;
  213. /* Extremely simple low-pass filter averages roughly the middle
  214. "half" of the CC pulse. (Times 14) */
  215. measured_level=
  216. samples[ind-13] + samples[ind-11] + samples[ind-9] +
  217. samples[ind-7] + samples[ind-5] + samples[ind-3] + samples[ind-1] +
  218. samples[ind+1] + samples[ind+3] + samples[ind+5] + samples[ind+7] +
  219. samples[ind+9] + samples[ind+11] + samples[ind+13];
  220. /* debugging code: */
  221. stats->nBitVals[18-i]= measured_level / 14;
  222. bits |= (measured_level > level);
  223. }
  224. /* Store the value of the 3 check bits; if this is valid CC, then
  225. bCheckBits should always be 4. */
  226. stats->bCheckBits = (bits & 7);
  227. /* Shift off the check bits. */
  228. bits >>= 3;
  229. /* Store the two bytes of decoded CC data. */
  230. dest[0] = bits & 0xff;
  231. dest[1] = bits >> 8;
  232. /* Our "quality" indicator runs from -1000...1030; divide this by 10
  233. to get -100...103 and truncate to get 0...100. */
  234. stats->nConfidence = quality/10;
  235. if (stats->nConfidence > 100) stats->nConfidence = 100;
  236. if (stats->nConfidence < 0) stats->nConfidence = 0;
  237. /* Record what we've computed about the signal. */
  238. stats->nFirstBit = (origin + 7*state->period) / CC_MULTIPLE;
  239. stats->nDC = level;
  240. /* Success! */
  241. return 0;
  242. }