2010-04-19 15:39:03 +00:00
|
|
|
#include "private/ftdm_core.h"
|
2007-06-05 18:27:51 +00:00
|
|
|
#include "fsk.h"
|
|
|
|
#include "uart.h"
|
|
|
|
|
|
|
|
|
2007-06-08 23:42:28 +00:00
|
|
|
|
2007-06-05 18:27:51 +00:00
|
|
|
static void fsk_byte_handler (void *x, int data)
|
|
|
|
{
|
2010-01-15 19:22:49 +00:00
|
|
|
ftdm_fsk_data_state_t *state = (ftdm_fsk_data_state_t *) x;
|
2007-06-06 15:59:38 +00:00
|
|
|
uint8_t byte = (uint8_t)data;
|
2007-06-05 18:27:51 +00:00
|
|
|
|
|
|
|
top:
|
|
|
|
|
|
|
|
if (state->init == 3) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (state->dlen) {
|
|
|
|
goto add_byte;
|
|
|
|
}
|
2008-02-18 16:15:24 +00:00
|
|
|
|
2007-06-05 18:27:51 +00:00
|
|
|
if (state->bpos == 1) {
|
|
|
|
state->blen = byte;
|
|
|
|
|
|
|
|
if ((uint32_t)(state->dlen = state->bpos + byte + 2) > state->bufsize) {
|
|
|
|
state->dlen = state->bufsize;
|
|
|
|
}
|
|
|
|
goto top;
|
|
|
|
}
|
|
|
|
|
|
|
|
add_byte:
|
|
|
|
|
|
|
|
if (state->bpos <= state->dlen) {
|
|
|
|
state->buf[state->bpos++] = byte;
|
|
|
|
} else {
|
|
|
|
state->init = 3;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
FT_DECLARE(ftdm_status_t) ftdm_fsk_data_init(ftdm_fsk_data_state_t *state, uint8_t *data, uint32_t datalen)
|
2007-06-08 23:42:28 +00:00
|
|
|
{
|
|
|
|
memset(state, 0, sizeof(*state));
|
|
|
|
state->buf = data;
|
|
|
|
state->bufsize = datalen;
|
|
|
|
state->bpos = 2;
|
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
return FTDM_SUCCESS;
|
2007-06-08 23:42:28 +00:00
|
|
|
}
|
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
FT_DECLARE(ftdm_status_t) ftdm_fsk_data_add_sdmf(ftdm_fsk_data_state_t *state, const char *date, char *number)
|
2007-06-08 23:42:28 +00:00
|
|
|
{
|
|
|
|
size_t dlen = strlen(date);
|
|
|
|
size_t nlen = strlen(number);
|
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
state->buf[0] = FTDM_CID_TYPE_SDMF;
|
2007-06-08 23:42:28 +00:00
|
|
|
memcpy(&state->buf[state->bpos], date, dlen);
|
|
|
|
state->bpos += dlen;
|
|
|
|
memcpy(&state->buf[state->bpos], number, nlen);
|
|
|
|
state->bpos += nlen;
|
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
return FTDM_SUCCESS;
|
2007-06-08 23:42:28 +00:00
|
|
|
}
|
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
FT_DECLARE(ftdm_status_t) ftdm_fsk_data_add_mdmf(ftdm_fsk_data_state_t *state, ftdm_mdmf_type_t type, const uint8_t *data, uint32_t datalen)
|
2007-06-08 23:42:28 +00:00
|
|
|
{
|
2010-01-15 19:22:49 +00:00
|
|
|
state->buf[0] = FTDM_CID_TYPE_MDMF;
|
2007-06-08 23:42:28 +00:00
|
|
|
state->buf[state->bpos++] = type;
|
2007-06-09 16:25:39 +00:00
|
|
|
state->buf[state->bpos++] = (uint8_t)datalen;
|
2007-06-08 23:42:28 +00:00
|
|
|
memcpy(&state->buf[state->bpos], data, datalen);
|
|
|
|
state->bpos += datalen;
|
2010-01-15 19:22:49 +00:00
|
|
|
return FTDM_SUCCESS;
|
2007-06-08 23:42:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
FT_DECLARE(ftdm_status_t) ftdm_fsk_data_add_checksum(ftdm_fsk_data_state_t *state)
|
2007-06-08 23:42:28 +00:00
|
|
|
{
|
|
|
|
uint32_t i;
|
|
|
|
uint8_t check = 0;
|
|
|
|
|
2007-06-09 16:25:39 +00:00
|
|
|
state->buf[1] = (uint8_t)(state->bpos - 2);
|
2007-06-08 23:42:28 +00:00
|
|
|
|
|
|
|
for (i = 0; i < state->bpos; i++) {
|
2007-06-09 16:25:39 +00:00
|
|
|
check = check + state->buf[i];
|
2007-06-08 23:42:28 +00:00
|
|
|
}
|
|
|
|
|
2009-03-06 18:17:15 +00:00
|
|
|
state->checksum = state->buf[state->bpos] = (uint8_t)(256 - check);
|
2007-06-08 23:42:28 +00:00
|
|
|
state->bpos++;
|
|
|
|
|
|
|
|
state->dlen = state->bpos;
|
|
|
|
state->blen = state->buf[1];
|
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
return FTDM_SUCCESS;
|
2007-06-08 23:42:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
FT_DECLARE(ftdm_status_t) ftdm_fsk_data_parse(ftdm_fsk_data_state_t *state, ftdm_size_t *type, char **data, ftdm_size_t *len)
|
2007-06-05 18:27:51 +00:00
|
|
|
{
|
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
ftdm_size_t i;
|
2007-06-06 15:59:38 +00:00
|
|
|
int sum = 0;
|
2007-06-05 18:27:51 +00:00
|
|
|
|
|
|
|
top:
|
|
|
|
|
|
|
|
if (state->checksum != 0 || state->ppos >= state->dlen - 1) {
|
2010-01-15 19:22:49 +00:00
|
|
|
return FTDM_FAIL;
|
2007-06-05 18:27:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!state->ppos) {
|
|
|
|
for(i = 0; i < state->bpos; i++) {
|
|
|
|
sum += state->buf[i];
|
|
|
|
}
|
|
|
|
state->checksum = sum % 256;
|
|
|
|
state->ppos = 2;
|
2007-06-08 23:42:28 +00:00
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
if (state->buf[0] != FTDM_CID_TYPE_MDMF && state->buf[0] != FTDM_CID_TYPE_SDMF) {
|
2007-06-05 18:27:51 +00:00
|
|
|
state->checksum = -1;
|
|
|
|
}
|
|
|
|
goto top;
|
|
|
|
}
|
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
if (state->buf[0] == FTDM_CID_TYPE_SDMF) {
|
2007-06-05 18:27:51 +00:00
|
|
|
/* convert sdmf to mdmf so we don't need 2 parsers */
|
|
|
|
if (state->ppos == 2) {
|
|
|
|
*type = MDMF_DATETIME;
|
|
|
|
*len = 8;
|
|
|
|
} else {
|
|
|
|
if (state->buf[state->ppos] == 'P' || state->buf[state->ppos] == 'O') {
|
|
|
|
*type = MDMF_NO_NUM;
|
|
|
|
*len = 1;
|
|
|
|
} else {
|
|
|
|
*type = MDMF_PHONE_NUM;
|
|
|
|
*len = state->blen - 8;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
*data = (char *)&state->buf[state->ppos];
|
|
|
|
state->ppos += *len;
|
2010-01-15 19:22:49 +00:00
|
|
|
return FTDM_SUCCESS;
|
|
|
|
} else if (state->buf[0] == FTDM_CID_TYPE_MDMF) {
|
2007-06-05 18:27:51 +00:00
|
|
|
*type = state->buf[state->ppos++];
|
|
|
|
*len = state->buf[state->ppos++];
|
|
|
|
*data = (char *)&state->buf[state->ppos];
|
|
|
|
state->ppos += *len;
|
2010-01-15 19:22:49 +00:00
|
|
|
return FTDM_SUCCESS;
|
2007-06-05 18:27:51 +00:00
|
|
|
}
|
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
return FTDM_FAIL;
|
2007-06-05 18:27:51 +00:00
|
|
|
}
|
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
FT_DECLARE(ftdm_status_t) ftdm_fsk_demod_feed(ftdm_fsk_data_state_t *state, int16_t *data, ftdm_size_t samples)
|
2007-06-05 18:27:51 +00:00
|
|
|
{
|
|
|
|
uint32_t x;
|
|
|
|
int16_t *sp = data;
|
|
|
|
|
|
|
|
if (state->init == 3) {
|
2010-01-15 19:22:49 +00:00
|
|
|
return FTDM_FAIL;
|
2007-06-05 18:27:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for (x = 0; x < samples; x++) {
|
|
|
|
dsp_fsk_sample (state->fsk1200_handle, (double) *sp++ / 32767.0);
|
|
|
|
if (state->dlen && state->bpos >= state->dlen) {
|
|
|
|
state->init = 3;
|
2010-01-15 19:22:49 +00:00
|
|
|
return FTDM_FAIL;
|
2007-06-05 18:27:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
return FTDM_SUCCESS;
|
2007-06-05 18:27:51 +00:00
|
|
|
}
|
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
FT_DECLARE(ftdm_status_t) ftdm_fsk_demod_destroy(ftdm_fsk_data_state_t *state)
|
2007-06-05 18:27:51 +00:00
|
|
|
{
|
|
|
|
dsp_fsk_destroy(&state->fsk1200_handle);
|
|
|
|
memset(state, 0, sizeof(*state));
|
2010-01-15 19:22:49 +00:00
|
|
|
return FTDM_SUCCESS;
|
2007-06-05 18:27:51 +00:00
|
|
|
}
|
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
FT_DECLARE(int) ftdm_fsk_demod_init(ftdm_fsk_data_state_t *state, int rate, uint8_t *buf, ftdm_size_t bufsize)
|
2007-06-05 18:27:51 +00:00
|
|
|
{
|
|
|
|
|
|
|
|
dsp_fsk_attr_t fsk1200_attr;
|
|
|
|
|
|
|
|
if (state->fsk1200_handle) {
|
|
|
|
dsp_fsk_destroy(&state->fsk1200_handle);
|
|
|
|
}
|
|
|
|
|
2007-06-15 08:16:38 +00:00
|
|
|
memset(state, 0, sizeof(*state));
|
|
|
|
memset(buf, 0, bufsize);
|
2007-06-05 18:27:51 +00:00
|
|
|
state->buf = buf;
|
|
|
|
state->bufsize = bufsize;
|
|
|
|
|
|
|
|
dsp_fsk_attr_init (&fsk1200_attr);
|
|
|
|
dsp_fsk_attr_set_samplerate (&fsk1200_attr, rate);
|
|
|
|
dsp_fsk_attr_set_bytehandler (&fsk1200_attr, fsk_byte_handler, state);
|
|
|
|
state->fsk1200_handle = dsp_fsk_create (&fsk1200_attr);
|
|
|
|
|
|
|
|
if (state->fsk1200_handle == NULL) {
|
2010-01-15 19:22:49 +00:00
|
|
|
return FTDM_FAIL;
|
2007-06-05 18:27:51 +00:00
|
|
|
}
|
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
return FTDM_SUCCESS;
|
2007-06-05 18:27:51 +00:00
|
|
|
}
|
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
FT_DECLARE(ftdm_size_t) ftdm_fsk_modulator_generate_bit(ftdm_fsk_modulator_t *fsk_trans, int8_t bit, int16_t *buf, ftdm_size_t buflen)
|
2007-06-08 23:42:28 +00:00
|
|
|
{
|
2010-01-15 19:22:49 +00:00
|
|
|
ftdm_size_t i;
|
2007-06-08 23:42:28 +00:00
|
|
|
|
|
|
|
for(i = 0 ; i < buflen; i++) {
|
|
|
|
fsk_trans->bit_accum += fsk_trans->bit_factor;
|
2010-01-15 19:22:49 +00:00
|
|
|
if (fsk_trans->bit_accum >= FTDM_FSK_MOD_FACTOR) {
|
|
|
|
fsk_trans->bit_accum -= (FTDM_FSK_MOD_FACTOR + fsk_trans->bit_factor);
|
2007-06-08 23:42:28 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
buf[i] = teletone_dds_state_modulate_sample(&fsk_trans->dds, bit);
|
|
|
|
}
|
|
|
|
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
FT_DECLARE(int32_t) ftdm_fsk_modulator_generate_carrier_bits(ftdm_fsk_modulator_t *fsk_trans, uint32_t bits)
|
2007-06-08 23:42:28 +00:00
|
|
|
{
|
|
|
|
uint32_t i = 0;
|
2010-01-15 19:22:49 +00:00
|
|
|
ftdm_size_t r = 0;
|
2007-06-09 16:25:39 +00:00
|
|
|
int8_t bit = 1;
|
2007-06-08 23:42:28 +00:00
|
|
|
|
|
|
|
for (i = 0; i < bits; i++) {
|
2010-01-15 19:22:49 +00:00
|
|
|
if ((r = ftdm_fsk_modulator_generate_bit(fsk_trans, bit, fsk_trans->sample_buffer, sizeof(fsk_trans->sample_buffer) / 2))) {
|
|
|
|
if (fsk_trans->write_sample_callback(fsk_trans->sample_buffer, r, fsk_trans->user_data) != FTDM_SUCCESS) {
|
2007-06-09 00:56:53 +00:00
|
|
|
break;
|
|
|
|
}
|
2007-06-08 23:42:28 +00:00
|
|
|
} else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
FT_DECLARE(void) ftdm_fsk_modulator_generate_chan_sieze(ftdm_fsk_modulator_t *fsk_trans)
|
2007-06-08 23:42:28 +00:00
|
|
|
{
|
|
|
|
uint32_t i = 0;
|
2010-01-15 19:22:49 +00:00
|
|
|
ftdm_size_t r = 0;
|
2007-06-09 16:25:39 +00:00
|
|
|
int8_t bit = 0;
|
2007-06-08 23:42:28 +00:00
|
|
|
|
|
|
|
for (i = 0; i < fsk_trans->chan_sieze_bits; i++) {
|
2010-01-15 19:22:49 +00:00
|
|
|
if ((r = ftdm_fsk_modulator_generate_bit(fsk_trans, bit, fsk_trans->sample_buffer, sizeof(fsk_trans->sample_buffer) / 2))) {
|
|
|
|
if (fsk_trans->write_sample_callback(fsk_trans->sample_buffer, r, fsk_trans->user_data) != FTDM_SUCCESS) {
|
2007-06-09 00:56:53 +00:00
|
|
|
break;
|
|
|
|
}
|
2007-06-08 23:42:28 +00:00
|
|
|
} else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
bit = !bit;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
FT_DECLARE(void) ftdm_fsk_modulator_send_data(ftdm_fsk_modulator_t *fsk_trans)
|
2007-06-08 23:42:28 +00:00
|
|
|
{
|
2010-01-15 19:22:49 +00:00
|
|
|
ftdm_size_t r = 0;
|
2007-06-08 23:42:28 +00:00
|
|
|
int8_t bit = 0;
|
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
while((bit = ftdm_bitstream_get_bit(&fsk_trans->bs)) > -1) {
|
|
|
|
if ((r = ftdm_fsk_modulator_generate_bit(fsk_trans, bit, fsk_trans->sample_buffer, sizeof(fsk_trans->sample_buffer) / 2))) {
|
|
|
|
if (fsk_trans->write_sample_callback(fsk_trans->sample_buffer, r, fsk_trans->user_data) != FTDM_SUCCESS) {
|
2007-06-09 00:56:53 +00:00
|
|
|
break;
|
|
|
|
}
|
2007-06-08 23:42:28 +00:00
|
|
|
} else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-01-15 19:22:49 +00:00
|
|
|
FT_DECLARE(ftdm_status_t) ftdm_fsk_modulator_init(ftdm_fsk_modulator_t *fsk_trans,
|
2007-06-08 23:42:28 +00:00
|
|
|
fsk_modem_types_t modem_type,
|
|
|
|
uint32_t sample_rate,
|
2010-01-15 19:22:49 +00:00
|
|
|
ftdm_fsk_data_state_t *fsk_data,
|
2007-06-08 23:42:28 +00:00
|
|
|
float db_level,
|
|
|
|
uint32_t carrier_bits_start,
|
|
|
|
uint32_t carrier_bits_stop,
|
|
|
|
uint32_t chan_sieze_bits,
|
2010-01-15 19:22:49 +00:00
|
|
|
ftdm_fsk_write_sample_t write_sample_callback,
|
2007-06-08 23:42:28 +00:00
|
|
|
void *user_data)
|
|
|
|
{
|
|
|
|
memset(fsk_trans, 0, sizeof(*fsk_trans));
|
|
|
|
fsk_trans->modem_type = modem_type;
|
|
|
|
teletone_dds_state_set_tone(&fsk_trans->dds, fsk_modem_definitions[fsk_trans->modem_type].freq_space, sample_rate, 0);
|
|
|
|
teletone_dds_state_set_tone(&fsk_trans->dds, fsk_modem_definitions[fsk_trans->modem_type].freq_mark, sample_rate, 1);
|
2010-01-15 19:22:49 +00:00
|
|
|
fsk_trans->bit_factor = (uint32_t)((fsk_modem_definitions[fsk_trans->modem_type].baud_rate * FTDM_FSK_MOD_FACTOR) / (float)sample_rate);
|
2007-06-09 00:56:53 +00:00
|
|
|
fsk_trans->samples_per_bit = (uint32_t) (sample_rate / fsk_modem_definitions[fsk_trans->modem_type].baud_rate);
|
2007-06-09 16:25:39 +00:00
|
|
|
fsk_trans->est_bytes = (int32_t)(((fsk_data->dlen * 10) + carrier_bits_start + carrier_bits_stop + chan_sieze_bits) * ((fsk_trans->samples_per_bit + 1) * 2));
|
2007-06-08 23:42:28 +00:00
|
|
|
fsk_trans->bit_accum = 0;
|
|
|
|
fsk_trans->fsk_data = fsk_data;
|
|
|
|
teletone_dds_state_set_tx_level(&fsk_trans->dds, db_level);
|
2010-01-15 19:22:49 +00:00
|
|
|
ftdm_bitstream_init(&fsk_trans->bs, fsk_trans->fsk_data->buf, (uint32_t)fsk_trans->fsk_data->dlen, FTDM_ENDIAN_BIG, 1);
|
2007-06-08 23:42:28 +00:00
|
|
|
fsk_trans->carrier_bits_start = carrier_bits_start;
|
|
|
|
fsk_trans->carrier_bits_stop = carrier_bits_stop;
|
|
|
|
fsk_trans->chan_sieze_bits = chan_sieze_bits;
|
|
|
|
fsk_trans->write_sample_callback = write_sample_callback;
|
|
|
|
fsk_trans->user_data = user_data;
|
2010-01-15 19:22:49 +00:00
|
|
|
return FTDM_SUCCESS;
|
2007-06-08 23:42:28 +00:00
|
|
|
}
|
2007-06-05 18:27:51 +00:00
|
|
|
|