fskmodem.c 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307
  1. /*
  2. * Asterisk -- An open source telephony toolkit.
  3. *
  4. * Copyright (C) 1999 - 2005, Digium, Inc.
  5. *
  6. * Mark Spencer <markster@digium.com>
  7. *
  8. * Includes code and algorithms from the Zapata library.
  9. *
  10. * See http://www.asterisk.org for more information about
  11. * the Asterisk project. Please do not directly contact
  12. * any of the maintainers of this project for assistance;
  13. * the project provides a web site, mailing lists and IRC
  14. * channels for your use.
  15. *
  16. * This program is free software, distributed under the terms of
  17. * the GNU General Public License Version 2. See the LICENSE file
  18. * at the top of the source tree.
  19. */
  20. /*! \file
  21. *
  22. * \brief FSK Modulator/Demodulator
  23. *
  24. */
  25. #include <stdio.h>
  26. #include "asterisk.h"
  27. ASTERISK_FILE_VERSION(__FILE__, "$Revision$")
  28. #include "asterisk/fskmodem.h"
  29. #define NBW 2
  30. #define BWLIST {75,800}
  31. #define NF 6
  32. #define FLIST {1400,1800,1200,2200,1300,2100}
  33. #define STATE_SEARCH_STARTBIT 0
  34. #define STATE_SEARCH_STARTBIT2 1
  35. #define STATE_SEARCH_STARTBIT3 2
  36. #define STATE_GET_BYTE 3
  37. static inline float get_sample(short **buffer, int *len)
  38. {
  39. float retval;
  40. retval = (float) **buffer / 256;
  41. (*buffer)++;
  42. (*len)--;
  43. return retval;
  44. }
  45. #define GET_SAMPLE get_sample(&buffer, len)
  46. /* Coeficientes para filtros de entrada */
  47. /* Tabla de coeficientes, generada a partir del programa "mkfilter" */
  48. /* Formato: coef[IDX_FREC][IDX_BW][IDX_COEF] */
  49. /* IDX_COEF=0 => 1/GAIN */
  50. /* IDX_COEF=1-6 => Coeficientes y[n] */
  51. static double coef_in[NF][NBW][8]={
  52. #include "coef_in.h"
  53. };
  54. /* Coeficientes para filtro de salida */
  55. /* Tabla de coeficientes, generada a partir del programa "mkfilter" */
  56. /* Formato: coef[IDX_BW][IDX_COEF] */
  57. /* IDX_COEF=0 => 1/GAIN */
  58. /* IDX_COEF=1-6 => Coeficientes y[n] */
  59. static double coef_out[NBW][8]={
  60. #include "coef_out.h"
  61. };
  62. /*! Filtro pasa-banda para frecuencia de MARCA */
  63. static inline float filtroM(fsk_data *fskd,float in)
  64. {
  65. int i,j;
  66. double s;
  67. double *pc;
  68. pc=&coef_in[fskd->f_mark_idx][fskd->bw][0];
  69. fskd->fmxv[(fskd->fmp+6)&7]=in*(*pc++);
  70. s=(fskd->fmxv[(fskd->fmp+6)&7] - fskd->fmxv[fskd->fmp]) + 3 * (fskd->fmxv[(fskd->fmp+2)&7] - fskd->fmxv[(fskd->fmp+4)&7]);
  71. for (i=0,j=fskd->fmp;i<6;i++,j++) s+=fskd->fmyv[j&7]*(*pc++);
  72. fskd->fmyv[j&7]=s;
  73. fskd->fmp++; fskd->fmp&=7;
  74. return s;
  75. }
  76. /*! Filtro pasa-banda para frecuencia de ESPACIO */
  77. static inline float filtroS(fsk_data *fskd,float in)
  78. {
  79. int i,j;
  80. double s;
  81. double *pc;
  82. pc=&coef_in[fskd->f_space_idx][fskd->bw][0];
  83. fskd->fsxv[(fskd->fsp+6)&7]=in*(*pc++);
  84. s=(fskd->fsxv[(fskd->fsp+6)&7] - fskd->fsxv[fskd->fsp]) + 3 * (fskd->fsxv[(fskd->fsp+2)&7] - fskd->fsxv[(fskd->fsp+4)&7]);
  85. for (i=0,j=fskd->fsp;i<6;i++,j++) s+=fskd->fsyv[j&7]*(*pc++);
  86. fskd->fsyv[j&7]=s;
  87. fskd->fsp++; fskd->fsp&=7;
  88. return s;
  89. }
  90. /*! Filtro pasa-bajos para datos demodulados */
  91. static inline float filtroL(fsk_data *fskd,float in)
  92. {
  93. int i,j;
  94. double s;
  95. double *pc;
  96. pc=&coef_out[fskd->bw][0];
  97. fskd->flxv[(fskd->flp + 6) & 7]=in * (*pc++);
  98. s= (fskd->flxv[fskd->flp] + fskd->flxv[(fskd->flp+6)&7]) +
  99. 6 * (fskd->flxv[(fskd->flp+1)&7] + fskd->flxv[(fskd->flp+5)&7]) +
  100. 15 * (fskd->flxv[(fskd->flp+2)&7] + fskd->flxv[(fskd->flp+4)&7]) +
  101. 20 * fskd->flxv[(fskd->flp+3)&7];
  102. for (i=0,j=fskd->flp;i<6;i++,j++) s+=fskd->flyv[j&7]*(*pc++);
  103. fskd->flyv[j&7]=s;
  104. fskd->flp++; fskd->flp&=7;
  105. return s;
  106. }
  107. static inline int demodulador(fsk_data *fskd, float *retval, float x)
  108. {
  109. float xS,xM;
  110. fskd->cola_in[fskd->pcola]=x;
  111. xS=filtroS(fskd,x);
  112. xM=filtroM(fskd,x);
  113. fskd->cola_filtro[fskd->pcola]=xM-xS;
  114. x=filtroL(fskd,xM*xM - xS*xS);
  115. fskd->cola_demod[fskd->pcola++]=x;
  116. fskd->pcola &= (NCOLA-1);
  117. *retval = x;
  118. return(0);
  119. }
  120. static int get_bit_raw(fsk_data *fskd, short *buffer, int *len)
  121. {
  122. /* Esta funcion implementa un DPLL para sincronizarse con los bits */
  123. float x,spb,spb2,ds;
  124. int f;
  125. spb=fskd->spb;
  126. if (fskd->spb == 7) spb = 8000.0 / 1200.0;
  127. ds=spb/32.;
  128. spb2=spb/2.;
  129. for (f=0;;){
  130. if (demodulador(fskd,&x, GET_SAMPLE)) return(-1);
  131. if ((x*fskd->x0)<0) { /* Transicion */
  132. if (!f) {
  133. if (fskd->cont<(spb2)) fskd->cont+=ds; else fskd->cont-=ds;
  134. f=1;
  135. }
  136. }
  137. fskd->x0=x;
  138. fskd->cont+=1.;
  139. if (fskd->cont>spb) {
  140. fskd->cont-=spb;
  141. break;
  142. }
  143. }
  144. f=(x>0)?0x80:0;
  145. return(f);
  146. }
  147. int fsk_serie(fsk_data *fskd, short *buffer, int *len, int *outbyte)
  148. {
  149. int a;
  150. int i,j,n1,r;
  151. int samples=0;
  152. int olen;
  153. int beginlen=*len;
  154. int beginlenx;
  155. switch(fskd->state) {
  156. /* Pick up where we left off */
  157. case STATE_SEARCH_STARTBIT2:
  158. goto search_startbit2;
  159. case STATE_SEARCH_STARTBIT3:
  160. goto search_startbit3;
  161. case STATE_GET_BYTE:
  162. goto getbyte;
  163. }
  164. /* Esperamos bit de start */
  165. do {
  166. /* this was jesus's nice, reasonable, working (at least with RTTY) code
  167. to look for the beginning of the start bit. Unfortunately, since TTY/TDD's
  168. just start sending a start bit with nothing preceding it at the beginning
  169. of a transmission (what a LOSING design), we cant do it this elegantly */
  170. /*
  171. if (demodulador(zap,&x1)) return(-1);
  172. for(;;) {
  173. if (demodulador(zap,&x2)) return(-1);
  174. if (x1>0 && x2<0) break;
  175. x1=x2;
  176. }
  177. */
  178. /* this is now the imprecise, losing, but functional code to detect the
  179. beginning of a start bit in the TDD sceanario. It just looks for sufficient
  180. level to maybe, perhaps, guess, maybe that its maybe the beginning of
  181. a start bit, perhaps. This whole thing stinks! */
  182. beginlenx=beginlen; /* just to avoid unused war warnings */
  183. if (demodulador(fskd,&fskd->x1,GET_SAMPLE)) return(-1);
  184. samples++;
  185. for(;;)
  186. {
  187. search_startbit2:
  188. if (*len <= 0) {
  189. fskd->state = STATE_SEARCH_STARTBIT2;
  190. return 0;
  191. }
  192. samples++;
  193. if (demodulador(fskd,&fskd->x2,GET_SAMPLE)) return(-1);
  194. #if 0
  195. printf("x2 = %5.5f ", fskd->x2);
  196. #endif
  197. if (fskd->x2 < -0.5) break;
  198. }
  199. search_startbit3:
  200. /* Esperamos 0.5 bits antes de usar DPLL */
  201. i=fskd->spb/2;
  202. if (*len < i) {
  203. fskd->state = STATE_SEARCH_STARTBIT3;
  204. return 0;
  205. }
  206. for(;i>0;i--) { if (demodulador(fskd,&fskd->x1,GET_SAMPLE)) return(-1);
  207. #if 0
  208. printf("x1 = %5.5f ", fskd->x1);
  209. #endif
  210. samples++; }
  211. /* x1 debe ser negativo (confirmación del bit de start) */
  212. } while (fskd->x1>0);
  213. fskd->state = STATE_GET_BYTE;
  214. getbyte:
  215. /* Need at least 80 samples (for 1200) or
  216. 1320 (for 45.5) to be sure we'll have a byte */
  217. if (fskd->nbit < 8) {
  218. if (*len < 1320)
  219. return 0;
  220. } else {
  221. if (*len < 80)
  222. return 0;
  223. }
  224. /* Leemos ahora los bits de datos */
  225. j=fskd->nbit;
  226. for (a=n1=0;j;j--) {
  227. olen = *len;
  228. i=get_bit_raw(fskd, buffer, len);
  229. buffer += (olen - *len);
  230. if (i == -1) return(-1);
  231. if (i) n1++;
  232. a>>=1; a|=i;
  233. }
  234. j=8-fskd->nbit;
  235. a>>=j;
  236. /* Leemos bit de paridad (si existe) y la comprobamos */
  237. if (fskd->paridad) {
  238. olen = *len;
  239. i=get_bit_raw(fskd, buffer, len);
  240. buffer += (olen - *len);
  241. if (i == -1) return(-1);
  242. if (i) n1++;
  243. if (fskd->paridad==1) { /* paridad=1 (par) */
  244. if (n1&1) a|=0x100; /* error */
  245. } else { /* paridad=2 (impar) */
  246. if (!(n1&1)) a|=0x100; /* error */
  247. }
  248. }
  249. /* Leemos bits de STOP. Todos deben ser 1 */
  250. for (j=fskd->nstop;j;j--) {
  251. r = get_bit_raw(fskd, buffer, len);
  252. if (r == -1) return(-1);
  253. if (!r) a|=0x200;
  254. }
  255. /* Por fin retornamos */
  256. /* Bit 8 : Error de paridad */
  257. /* Bit 9 : Error de Framming */
  258. *outbyte = a;
  259. fskd->state = STATE_SEARCH_STARTBIT;
  260. return 1;
  261. }