kopia lustrzana https://github.com/rs1729/RS
X50 xpos?
rodzic
4066826d01
commit
dbd2fc4d62
|
@ -0,0 +1,627 @@
|
|||
|
||||
/*
|
||||
C50
|
||||
(recommended: sample rate 48kHz)
|
||||
gcc c50dft.c -lm -o c50dft
|
||||
*/
|
||||
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
#include <complex.h>
|
||||
#include <math.h>
|
||||
|
||||
#ifdef CYGWIN
|
||||
#include <fcntl.h> // cygwin: _setmode()
|
||||
#include <io.h>
|
||||
#endif
|
||||
|
||||
// optional JSON "version"
|
||||
// (a) set global
|
||||
// gcc -DVERSION_JSN [-I<inc_dir>] ...
|
||||
#ifdef VERSION_JSN
|
||||
#include "version_jsn.h"
|
||||
#endif
|
||||
// or
|
||||
// (b) set local compiler option, e.g.
|
||||
// gcc -DVER_JSN_STR=\"0.0.2\" ...
|
||||
|
||||
|
||||
typedef unsigned char ui8_t;
|
||||
typedef unsigned int ui32_t;
|
||||
|
||||
static int option_verbose = 0,
|
||||
option_xor = 0,
|
||||
option_raw = 0,
|
||||
option_ptu = 0,
|
||||
option_json = 0,
|
||||
wavloaded = 0;
|
||||
|
||||
|
||||
typedef struct {
|
||||
//int frnr;
|
||||
int sn;
|
||||
int jahr; int monat; int tag;
|
||||
int std; int min; int sek;
|
||||
float lat; float lon; float alt;
|
||||
ui32_t raw_lat; ui32_t raw_lon; ui32_t raw_alt;
|
||||
unsigned chk;
|
||||
float T; float RH;
|
||||
int jsn_freq; // freq/kHz (SDR)
|
||||
} gpx_t;
|
||||
|
||||
static gpx_t gpx;
|
||||
|
||||
/* ------------------------------------------------------------------------------------ */
|
||||
|
||||
|
||||
#define BAUD_RATE 2400
|
||||
|
||||
static unsigned int sample_rate = 0;
|
||||
static int bits_sample = 0, channels = 0;
|
||||
//float samples_per_bit = 0;
|
||||
|
||||
static int findstr(char *buff, char *str, int pos) {
|
||||
int i;
|
||||
for (i = 0; i < 4; i++) {
|
||||
if (buff[(pos+i)%4] != str[i]) break;
|
||||
}
|
||||
return i;
|
||||
}
|
||||
|
||||
static int read_wav_header(FILE *fp) {
|
||||
char txt[4+1] = "\0\0\0\0";
|
||||
unsigned char dat[4];
|
||||
int byte, p=0;
|
||||
|
||||
if (fread(txt, 1, 4, fp) < 4) return -1;
|
||||
if (strncmp(txt, "RIFF", 4) && strncmp(txt, "RF64", 4)) return -1;
|
||||
|
||||
if (fread(txt, 1, 4, fp) < 4) return -1;
|
||||
// pos_WAVE = 8L
|
||||
if (fread(txt, 1, 4, fp) < 4) return -1;
|
||||
if (strncmp(txt, "WAVE", 4)) return -1;
|
||||
|
||||
// pos_fmt = 12L
|
||||
for ( ; ; ) {
|
||||
if ( (byte=fgetc(fp)) == EOF ) return -1;
|
||||
txt[p % 4] = byte;
|
||||
p++; if (p==4) p=0;
|
||||
if (findstr(txt, "fmt ", p) == 4) break;
|
||||
}
|
||||
if (fread(dat, 1, 4, fp) < 4) return -1;
|
||||
if (fread(dat, 1, 2, fp) < 2) return -1;
|
||||
|
||||
if (fread(dat, 1, 2, fp) < 2) return -1;
|
||||
channels = dat[0] + (dat[1] << 8);
|
||||
|
||||
if (fread(dat, 1, 4, fp) < 4) return -1;
|
||||
memcpy(&sample_rate, dat, 4); //sample_rate = dat[0]|(dat[1]<<8)|(dat[2]<<16)|(dat[3]<<24);
|
||||
|
||||
if (fread(dat, 1, 4, fp) < 4) return -1;
|
||||
if (fread(dat, 1, 2, fp) < 2) return -1;
|
||||
//byte = dat[0] + (dat[1] << 8);
|
||||
|
||||
if (fread(dat, 1, 2, fp) < 2) return -1;
|
||||
bits_sample = dat[0] + (dat[1] << 8);
|
||||
|
||||
// pos_dat = 36L + info
|
||||
for ( ; ; ) {
|
||||
if ( (byte=fgetc(fp)) == EOF ) return -1;
|
||||
txt[p % 4] = byte;
|
||||
p++; if (p==4) p=0;
|
||||
if (findstr(txt, "data", p) == 4) break;
|
||||
}
|
||||
if (fread(dat, 1, 4, fp) < 4) return -1;
|
||||
|
||||
|
||||
fprintf(stderr, "sample_rate: %d\n", sample_rate);
|
||||
fprintf(stderr, "bits : %d\n", bits_sample);
|
||||
fprintf(stderr, "channels : %d\n", channels);
|
||||
|
||||
if ((bits_sample != 8) && (bits_sample != 16) && (bits_sample != 32)) return -1;
|
||||
|
||||
if (sample_rate == 900001) sample_rate -= 1;
|
||||
|
||||
//samples_per_bit = sample_rate/(float)BAUD_RATE;
|
||||
//fprintf(stderr, "samples/bit: %.2f\n", samples_per_bit);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static unsigned int sample_count;
|
||||
|
||||
static int f32read_sample(FILE *fp, float *s) {
|
||||
int i;
|
||||
unsigned int word = 0;
|
||||
short *b = (short*)&word;
|
||||
float *f = (float*)&word;
|
||||
|
||||
for (i = 0; i < channels; i++) {
|
||||
|
||||
if (fread( &word, bits_sample/8, 1, fp) != 1) return EOF;
|
||||
|
||||
if (i == 0) { // i = 0: links bzw. mono
|
||||
//if (bits_sample == 8) sint = b-128; // 8bit: 00..FF, centerpoint 0x80=128
|
||||
//if (bits_sample == 16) sint = (short)b;
|
||||
|
||||
if (bits_sample == 32) {
|
||||
*s = *f;
|
||||
}
|
||||
else {
|
||||
if (bits_sample == 8) { *b -= 128; }
|
||||
*s = *b/(float)(1<<bits_sample);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* ------------------------------------------------------------------------------------ */
|
||||
|
||||
|
||||
#define LOG2N 6 // 2^7 = 128 = N
|
||||
#define N 64 // 128 Vielfaches von 22 oder 10 unten
|
||||
#define WLEN 40 // (2*(48000/BAUDRATE))
|
||||
|
||||
#define BITS 10 // (8N1: 0 bbbbbbbb 1)
|
||||
#define LEN_BITFRAME (9*BITS)
|
||||
#define HEADLEN 20
|
||||
|
||||
static char header[] = "00000000010111111111"; // 0x00 0xFF
|
||||
static char buf[HEADLEN+1] = "x";
|
||||
static int bufpos = -1;
|
||||
|
||||
static int bitpos;
|
||||
static ui8_t bits[LEN_BITFRAME+20] = { 0, 0, 0, 0, 0, 0, 0, 0, 0, 1,
|
||||
0, 1, 1, 1, 1, 1, 1, 1, 1, 1};
|
||||
static ui8_t bytes[LEN_BITFRAME/BITS+2];
|
||||
|
||||
static float complex w[N], ew[N*N];
|
||||
|
||||
static int ptr;
|
||||
static float Hann[N], buffer[N+1];
|
||||
|
||||
|
||||
static void init_dft() {
|
||||
int i, k, n;
|
||||
|
||||
for (i = 0; i < N; i++) Hann[i] = 0;
|
||||
for (i = 0; i < WLEN; i++) Hann[i] = 0.5 * (1 - cos( 2 * M_PI * i / (float)(WLEN-1) ) );
|
||||
//Hann[i+(N-1-WLEN)/2] = 0.5 * (1 - cos( 2 * M_PI * i / (float)(WLEN-1) ) );
|
||||
|
||||
for (k = 0; k < N; k++) {
|
||||
w[k] = -I*2*M_PI * k / (float)N;
|
||||
for (n = 0; n < N; n++) {
|
||||
ew[k*n] = cexp( w[k] * n );
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
static float dft_k(int k) {
|
||||
int n;
|
||||
float complex Zk;
|
||||
|
||||
Zk = 0;
|
||||
for (n = 0; n < WLEN; n++) { //Hann[WLEN:N]=0
|
||||
// Hann[n]*buffer[(ptr + n + 1)%N] * ew[k*n];
|
||||
Zk += Hann[n]*buffer[(ptr + n + 1)&(N-1)] * ew[k*n]; // N=2^6=64
|
||||
}
|
||||
return cabs(Zk);
|
||||
}
|
||||
|
||||
static float freq2bin(int f) {
|
||||
return f * N / (float)sample_rate;
|
||||
}
|
||||
|
||||
static int bin2freq(int k) {
|
||||
return sample_rate * k / N;
|
||||
}
|
||||
|
||||
/* ------------------------------------------------------------------------------------ */
|
||||
|
||||
|
||||
static void inc_bufpos() {
|
||||
bufpos = (bufpos+1) % HEADLEN;
|
||||
}
|
||||
|
||||
static int compare() {
|
||||
int i=0, j = bufpos;
|
||||
|
||||
while (i < HEADLEN) {
|
||||
if (j < 0) j = HEADLEN-1;
|
||||
if (buf[j] != header[HEADLEN-1-i]) break;
|
||||
j--;
|
||||
i++;
|
||||
}
|
||||
return i;
|
||||
}
|
||||
|
||||
static int bits2bytes8N1(ui8_t bits[], ui8_t bytes[], int n) {
|
||||
int i, j, byteval=0, d=1;
|
||||
|
||||
for (j = 0; j < n; j++) {
|
||||
byteval=0; d=1;
|
||||
for (i = 1; i < BITS-1; i++) { // little endian
|
||||
/* for (i = 7; i >= 0; i--) { // big endian */
|
||||
if (bits[BITS*j+i] == 1) byteval += d;
|
||||
else /*(bits[BITS*j+i] == 0)*/ byteval += 0;
|
||||
d <<= 1;
|
||||
}
|
||||
bytes[j] = byteval;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int xor_base = 0x7CA00000 ^ 0x2EFCF; // TODO
|
||||
static int xmask = 0;
|
||||
|
||||
static void printGPX() {
|
||||
int i;
|
||||
|
||||
if (gpx.sn) printf("( %d ) ", gpx.sn);
|
||||
printf(" %04d-%02d-%02d", gpx.jahr, gpx.monat, gpx.tag);
|
||||
printf(" %02d:%02d:%02d", gpx.std, gpx.min, gpx.sek);
|
||||
printf(" ");
|
||||
printf(" lat: %.5f", gpx.lat);
|
||||
printf(" lon: %.5f", gpx.lon);
|
||||
printf(" alt: %.1f", gpx.alt);
|
||||
|
||||
if (option_ptu && (gpx.T > -273.0 || gpx.RH > -0.5)) {
|
||||
printf(" ");
|
||||
if (gpx.T > -273.0) printf(" T=%.1fC", gpx.T);
|
||||
if (gpx.RH > -0.5) printf(" RH=%.0f%%", gpx.RH);
|
||||
}
|
||||
|
||||
if (option_verbose) {
|
||||
printf(" # ");
|
||||
for (i = 0; i < 5; i++) printf("%d", (gpx.chk>>i)&1);
|
||||
if (option_ptu) for (i = 6; i < 8; i++) printf("%d", (gpx.chk>>i)&1);
|
||||
}
|
||||
|
||||
printf("\n");
|
||||
}
|
||||
|
||||
static void printJSON() {
|
||||
// UTC or GPS time ?
|
||||
char *ver_jsn = NULL;
|
||||
char json_sonde_id[] = "C50-xxxx\0\0\0\0\0\0\0";
|
||||
if (gpx.sn) {
|
||||
sprintf(json_sonde_id, "C50-%u", gpx.sn);
|
||||
}
|
||||
printf("{ \"type\": \"%s\"", "C50");
|
||||
printf(", \"id\": \"%s\", ", json_sonde_id);
|
||||
printf("\"datetime\": \"%04d-%02d-%02dT%02d:%02d:%02dZ\", \"lat\": %.5f, \"lon\": %.5f, \"alt\": %.1f",
|
||||
gpx.jahr, gpx.monat, gpx.tag, gpx.std, gpx.min, gpx.sek, gpx.lat, gpx.lon, gpx.alt);
|
||||
if (option_ptu && (gpx.T > -273.0 || gpx.RH > -0.5)) {
|
||||
if (gpx.T > -273.0) printf(", \"temp\": %.1f", gpx.T);
|
||||
if (gpx.RH > -0.5) printf(", \"humidity\": %.1f", gpx.RH);
|
||||
}
|
||||
if (gpx.jsn_freq > 0) {
|
||||
printf(", \"freq\": %d", gpx.jsn_freq);
|
||||
}
|
||||
|
||||
// Reference time/position
|
||||
printf(", \"ref_datetime\": \"%s\"", "UTC" ); // {"GPS", "UTC"} GPS-UTC=leap_sec
|
||||
printf(", \"ref_position\": \"%s\"", "MSL" ); // {"GPS", "MSL"} GPS=ellipsoid , MSL=geoid
|
||||
|
||||
#ifdef VER_JSN_STR
|
||||
ver_jsn = VER_JSN_STR;
|
||||
#endif
|
||||
if (ver_jsn && *ver_jsn != '\0') printf(", \"version\": \"%s\"", ver_jsn);
|
||||
printf(" }\n");
|
||||
//printf("\n");
|
||||
}
|
||||
|
||||
// Chechsum Fletcher16
|
||||
static unsigned check2(ui8_t *bytes, int len) {
|
||||
int sum1, sum2;
|
||||
int i;
|
||||
|
||||
sum1 = 0;
|
||||
sum2 = 0;
|
||||
for (i = 0; i < len; i++) {
|
||||
sum1 += bytes[i];
|
||||
sum2 += (len-i)*bytes[i];
|
||||
}
|
||||
sum1 = sum1 & 0xFF;
|
||||
sum2 = (-1-sum2) & 0xFF; // = (~sum2) & 0xFF;
|
||||
|
||||
return sum2 | (sum1<<8);
|
||||
}
|
||||
/* // equivalent
|
||||
static unsigned check16(ui8_t *bytes, int len) {
|
||||
unsigned sum1, sum2;
|
||||
int i;
|
||||
sum1 = sum2 = 0;
|
||||
for (i = 0; i < len; i++) {
|
||||
sum1 = (sum1 + bytes[i]) % 0x100;
|
||||
sum2 = (sum2 + sum1) % 0x100;
|
||||
}
|
||||
sum2 = (~sum2) & 0xFF; // 1's complement
|
||||
return sum2 | (sum1<<8);
|
||||
}
|
||||
*/
|
||||
|
||||
static float NMEAll2(int ll) { // NMEA GGA,GLL: ll/1e5=(D)DDMM.mmmm
|
||||
int deg = ll / 10000000;
|
||||
float min = (ll - deg*10000000)/1e5;
|
||||
return deg+min/60.0;
|
||||
}
|
||||
|
||||
static int evalBytes2() {
|
||||
int i, val = 0;
|
||||
ui8_t id = bytes[2];
|
||||
unsigned check;
|
||||
static unsigned int cnt_dat = -1, cnt_tim = -1,
|
||||
cnt_lat = -1, cnt_lon = -1, cnt_alt = -1,
|
||||
cnt_sn = -1,
|
||||
cnt_t3 = -1, cnt_rh = -1;
|
||||
|
||||
check = ((bytes[7]<<8)|bytes[8]) != check2(bytes+2, 5);
|
||||
|
||||
for (i = 0; i < 4; i++) val |= bytes[6-i] << (8*i);
|
||||
|
||||
if (id == 0x14 ) { // date
|
||||
int tag = val / 10000;
|
||||
int mon = (val-tag*10000) / 100;
|
||||
int jrz = val % 100;
|
||||
gpx.tag = tag;
|
||||
gpx.monat = mon;
|
||||
gpx.jahr = 2000+jrz;
|
||||
gpx.chk = (gpx.chk & ~(0x1<<0)) | (check<<0);
|
||||
if (check==0) cnt_dat = sample_count;
|
||||
}
|
||||
else if (id == 0x15 ) { // time (UTC)
|
||||
int std = val / 10000;
|
||||
int min = (val-std*10000) / 100;
|
||||
int sek = val % 100;
|
||||
gpx.std = std;
|
||||
gpx.min = min;
|
||||
gpx.sek = sek;
|
||||
gpx.chk = (gpx.chk & ~(0x1<<1)) | (check<<1);
|
||||
if (check==0) cnt_tim = sample_count;
|
||||
|
||||
// 2024-10-03, 2024-12-10: xpos?
|
||||
xmask = xor_base ^ val; // TODO
|
||||
}
|
||||
else if (id == 0x16 ) { // lat: wie NMEA mit Faktor 1e5 // <-> id==0x32
|
||||
gpx.raw_lat = val;
|
||||
if (option_xor) val ^= xmask;
|
||||
gpx.lat = NMEAll2(val);
|
||||
gpx.chk = (gpx.chk & ~(0x1<<2)) | (check<<2);
|
||||
if (check==0) cnt_lat = sample_count;
|
||||
}
|
||||
else if (id == 0x17 ) { // lon: wie NMEA mit Faktor 1e5 // <-> id==0x33
|
||||
gpx.raw_lon = val;
|
||||
if (option_xor) val ^= xmask;
|
||||
gpx.lon = NMEAll2(val);
|
||||
gpx.chk = (gpx.chk & ~(0x1<<3)) | (check<<3);
|
||||
if (check==0) cnt_lon = sample_count;
|
||||
}
|
||||
else if (id == 0x18 ) { // alt: decimeter (MSL) // <-> id==0x34
|
||||
gpx.raw_alt = val;
|
||||
if (option_xor) val ^= xmask;
|
||||
gpx.alt = val/10.0;
|
||||
gpx.chk = (gpx.chk & ~(0x1<<4)) | (check<<4);
|
||||
if (check==0) cnt_alt = sample_count;
|
||||
}
|
||||
else if (id == 0x64 ) { // serial number
|
||||
if (check==0) gpx.sn = val; // 16 bit
|
||||
//gpx.chk = (gpx.chk & ~(0x1<<15)) | (check<<15);
|
||||
//if (check==0) cnt_sn = sample_count;
|
||||
}
|
||||
|
||||
if (id == 0x18) {
|
||||
printGPX();
|
||||
if (option_json && check==0) {
|
||||
if ( ((cnt_dat|cnt_tim|cnt_lat|cnt_lon)&0x80000000)==0 &&
|
||||
cnt_alt - cnt_dat < sample_rate &&
|
||||
cnt_alt - cnt_tim < sample_rate &&
|
||||
cnt_alt - cnt_lat < sample_rate &&
|
||||
cnt_alt - cnt_lon < sample_rate )
|
||||
{
|
||||
if (cnt_alt - cnt_t3 > sample_rate) gpx.T = -273.15;
|
||||
if (cnt_alt - cnt_rh > sample_rate) gpx.RH = -1.0;
|
||||
printJSON();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// PTU floats
|
||||
if (id == 0x03) { // temperature
|
||||
float t = -273.15;
|
||||
memcpy(&t, &val, 4);
|
||||
if (t < -273.0 || t > 100.0) t = -273.15;
|
||||
gpx.T = t;
|
||||
gpx.chk = (gpx.chk & ~(0x1<<6)) | (check<<6);
|
||||
if (check==0) cnt_t3 = sample_count;
|
||||
}
|
||||
if (id == 0x10) { // rel. humidity
|
||||
float rh = -1.0;
|
||||
memcpy(&rh, &val, 4);
|
||||
if (rh < -0.4 || rh > 110.0) rh = -1.0;
|
||||
gpx.RH = rh;
|
||||
gpx.chk = (gpx.chk & ~(0x1<<7)) | (check<<7);
|
||||
if (check==0) cnt_rh = sample_count;
|
||||
}
|
||||
|
||||
return check;
|
||||
}
|
||||
|
||||
static void printRaw(int n) {
|
||||
int j;
|
||||
unsigned chkbyt = (bytes[7]<<8) | bytes[8];
|
||||
unsigned chksum = check2(bytes+2, 5);
|
||||
//if (chksum == chkbyt)
|
||||
{
|
||||
for (j = 0; j < LEN_BITFRAME; j++) {
|
||||
if (j%BITS == 1) printf(" ");
|
||||
if (j%BITS == 9) printf(" ");
|
||||
printf("%d", bits[j]);
|
||||
}
|
||||
printf(" : ");
|
||||
printf("%02X%02X ", bytes[0], bytes[1]);
|
||||
printf("%02X ", bytes[2]);
|
||||
printf("%02X%02X%02X%02X ", bytes[3], bytes[4], bytes[5], bytes[6]);
|
||||
printf("%02X%02X", bytes[7], bytes[8]); // chkbyt
|
||||
if (option_verbose) {
|
||||
printf(" # %04X", chksum);
|
||||
if (chksum == chkbyt) printf(" [OK]"); else printf(" [NO]");
|
||||
}
|
||||
printf("\n");
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
int main(int argc, char *argv[]) {
|
||||
|
||||
FILE *fp;
|
||||
char *fpname;
|
||||
int i, k0, k1;
|
||||
int bit = 8, bit0 = 8;
|
||||
int pos = 0, pos0 = 0;
|
||||
int header_found = 0;
|
||||
int bitlen; // sample_rate/BAUD_RATE
|
||||
int len;
|
||||
float k_f0, k_f1;
|
||||
float cb0, cb1;
|
||||
float s = 0.0;
|
||||
int cfreq = -1;
|
||||
|
||||
#ifdef CYGWIN
|
||||
_setmode(fileno(stdin), _O_BINARY); // _setmode(_fileno(stdin), _O_BINARY);
|
||||
#endif
|
||||
setbuf(stdout, NULL);
|
||||
|
||||
|
||||
fpname = argv[0];
|
||||
++argv;
|
||||
while ((*argv) && (!wavloaded)) {
|
||||
if ( (strcmp(*argv, "-h") == 0) || (strcmp(*argv, "--help") == 0) ) {
|
||||
fprintf(stderr, "%s [options] audio.wav\n", fpname);
|
||||
fprintf(stderr, " options:\n");
|
||||
fprintf(stderr, " -v, --verbose\n");
|
||||
fprintf(stderr, " -r, --raw\n");
|
||||
return 0;
|
||||
}
|
||||
else if ( (strcmp(*argv, "-v") == 0) || (strcmp(*argv, "--verbose") == 0) ) {
|
||||
option_verbose = 1;
|
||||
}
|
||||
else if ( (strcmp(*argv, "-r") == 0) || (strcmp(*argv, "--raw") == 0) ) {
|
||||
option_raw = 1;
|
||||
}
|
||||
else if ( (strcmp(*argv, "--xor") == 0) ) {
|
||||
option_xor = 1;
|
||||
}
|
||||
else if ( (strcmp(*argv, "--ptu") == 0) ) {
|
||||
option_ptu = 1;
|
||||
}
|
||||
else if ( (strcmp(*argv, "--json") == 0) ) {
|
||||
option_verbose = 1;
|
||||
option_json = 1;
|
||||
}
|
||||
else if ( (strcmp(*argv, "--jsn_cfq") == 0) ) {
|
||||
int frq = -1; // center frequency / Hz
|
||||
++argv;
|
||||
if (*argv) frq = atoi(*argv); else return -1;
|
||||
if (frq < 300000000) frq = -1;
|
||||
cfreq = frq;
|
||||
}
|
||||
else {
|
||||
fp = fopen(*argv, "rb");
|
||||
if (fp == NULL) {
|
||||
fprintf(stderr, "error: open %s\n", *argv);
|
||||
return -1;
|
||||
}
|
||||
wavloaded = 1;
|
||||
}
|
||||
++argv;
|
||||
}
|
||||
if (!wavloaded) fp = stdin;
|
||||
|
||||
|
||||
gpx.jsn_freq = 0;
|
||||
if (cfreq > 0) gpx.jsn_freq = (cfreq+500)/1000;
|
||||
|
||||
i = read_wav_header(fp);
|
||||
if (i) {
|
||||
fclose(fp);
|
||||
return -1;
|
||||
}
|
||||
if ( sample_rate != 48000 ) {
|
||||
fprintf(stderr, "note: sample rate not 48000\n");
|
||||
}
|
||||
|
||||
|
||||
bitlen = sample_rate/BAUD_RATE;
|
||||
k_f0 = freq2bin(4700); // bit0: 4800Hz
|
||||
k_f1 = freq2bin(2900); // bit1: 3000Hz
|
||||
k0 = (int)(k_f0+.5);
|
||||
k1 = (int)(k_f1+.5);
|
||||
|
||||
init_dft();
|
||||
|
||||
ptr = -1; sample_count = -1;
|
||||
while (f32read_sample(fp, &s) != EOF) {
|
||||
|
||||
ptr++;
|
||||
sample_count++;
|
||||
if (ptr == N) ptr = 0;
|
||||
buffer[ptr] = s;
|
||||
|
||||
if (sample_count < N) continue;
|
||||
|
||||
cb0 = dft_k(k0);
|
||||
cb1 = dft_k(k1);
|
||||
|
||||
bit = (cb0 > cb1) ? 0 : 1;
|
||||
|
||||
if (bit != bit0) {
|
||||
|
||||
pos0 = pos;
|
||||
pos = sample_count; //sample_count-(N-1)/2
|
||||
|
||||
len = (pos-pos0+bitlen/2)/bitlen; //(pos-pos0)/(float)bitlen + 0.5;
|
||||
for (i = 0; i < len; i++) {
|
||||
inc_bufpos();
|
||||
buf[bufpos] = 0x30 + bit0;
|
||||
|
||||
if (!header_found) {
|
||||
if (compare() >= HEADLEN-1) {
|
||||
header_found = 1;
|
||||
for (bitpos = 0; bitpos < HEADLEN; bitpos++) bits[bitpos] = header[bitpos] & 0x1;
|
||||
}
|
||||
}
|
||||
else {
|
||||
bits[bitpos] = bit0;
|
||||
bitpos++;
|
||||
if (bitpos >= LEN_BITFRAME) {
|
||||
|
||||
bits2bytes8N1(bits, bytes, bitpos/BITS);
|
||||
|
||||
if (option_raw) {
|
||||
printRaw(bitpos/BITS);
|
||||
}
|
||||
else {
|
||||
evalBytes2();
|
||||
}
|
||||
|
||||
bitpos = 0;
|
||||
header_found = 0;
|
||||
}
|
||||
}
|
||||
}
|
||||
bit0 = bit;
|
||||
}
|
||||
|
||||
}
|
||||
printf("\n");
|
||||
|
||||
fclose(fp);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
Ładowanie…
Reference in New Issue