Initial revision

master
Lucjan Bryndza 2013-03-25 00:13:06 +01:00
commit 0fd4ad894e
7 zmienionych plików z 1786 dodań i 0 usunięć

36
Makefile 100644
Wyświetl plik

@ -0,0 +1,36 @@
# Automatic makefile for GNUARM (C/C++)
#Target binary file name
TARGET = psk31test
#Optimalization [0,1,2,3,s]
# 0 - none optimalization, s - size optimalization 3 - most optimized
OPT ?= 2
#Common flags
COMMON_FLAGS = -pipe -Wall -pedantic -Wextra -Wno-vla -I.
COMMON_FLAGS += -D__STDC_CONSTANT_MACROS -I./libpsk/include
#C compiler options
CFLAGS += $(COMMON_FLAGS)
CFLAGS += -std=gnu99
#C++ compiler options
CXXFLAGS += $(COMMON_FLAGS) -std=c++11
#LDflags libraries etc.
LDFLAGS += -lavformat -lavcodec -lz -lavutil
#Per file listing
LISTING = n
#Debug version
DEBUG ?= y
#Source C++ files
CPPSRC += $(wildcard *.cpp) $(wildcard libpsk/src/*.cpp)
include unix.mk

Wyświetl plik

@ -0,0 +1,162 @@
/*
* decoder.h
*
* Created on: 24-03-2013
* Author: lucck
*/
/* ------------------------------------------------------------------------- */
#ifndef LIBPSK_DECODER_HPP_
#define LIBPSK_DECODER_HPP_
/* ------------------------------------------------------------------------- */
#include <cstddef>
#include <array>
#include <complex>
#include "imd_calculator.hpp"
/* ------------------------------------------------------------------------- */
namespace ham {
namespace psk {
/* ------------------------------------------------------------------------- */
//Sample type variable
typedef short sample_type;
//Vector data
typedef std::array<std::pair<long, long>, 8> signal_vector_type;
//Sync vector data
typedef std::array<long, 16> sync_array_type;
//Squelch tresh type
typedef int sqelch_value_type;
//Sample rate type
typedef short samplerate_type;
/* ------------------------------------------------------------------------- */
class decoder {
//Make object noncopyable
decoder(const decoder&) = delete;
decoder& operator=(const decoder&) = delete;
static constexpr auto DEC4_LPFIR_LENGTH = 35;
static constexpr auto BITFIR_LENGTH = 65;
//Survivor states in decoder
struct survivor_states
{
survivor_states( double _path_distance = double(), long _bit_estimates = long() )
: path_distance(_path_distance), bit_estimates(_bit_estimates ) {}
double path_distance; // sum of all metrics for a given survivor path
long bit_estimates; // the bit pattern estimate associated with given survivor path
};
public:
enum class mode
{
bpsk, //BPSK
qpsku, //QPSK USB
qpskl //QPSK LSB
};
enum class baudrate
{
b31,
b63,
b125
};
enum class squelch_mode
{
slow,
fast
};
//Construct the decoder object
explicit decoder( samplerate_type sample_rate );
//Process input sample buffer
void operator()( sample_type* samples, std::size_t sample_size );
//Get signal vector
signal_vector_type get_signal_vector( ) const;
//Get sync array type
sync_array_type get_sync_data() const;
//Reset decoder
void reset();
//Set receive frequency
void set_frequency( int freq );
//Set detector mode
void set_mode( mode mode, baudrate rate );
//Set AFC limit
void set_afc_limit( int limit );
//Get current frequency
int get_frequency() const;
//Get signal level
int get_signal_level() const;
//Set squelch tresh
void set_squelch_tresh( sqelch_value_type tresh, squelch_mode mode );
void calc_quality( double angle );
private:
bool viterbi_decode( double newangle );
void calc_bit_filter( std::complex<double> samp );
void calc_agc( std::complex<double> samp );
void calc_freq_error( std::complex<double> IQ );
void calc_ffreq_error( std::complex<double> IQ );
void decode_symb( std::complex<double> newsamp );
bool symb_sync(std::complex<double> sample);
bool is_qpsk() const
{
return m_rx_mode != mode::bpsk;
}
private:
baudrate m_baudrate;
double m_vco_phz;
int m_afc_timer;
bool m_afc_capture_on;
int m_rx_frequency;
double m_nco_phzinc;
double m_afc_limit;
double m_afc_max;
double m_afc_min;
std::array<std::complex<double>, DEC4_LPFIR_LENGTH> m_que1;
std::array<std::complex<double>, DEC4_LPFIR_LENGTH> m_que2;
std::array<std::complex<double>, BITFIR_LENGTH> m_que3;
int m_fir1_state;
int m_fir2_state;
int m_fir3_state;
std::array<survivor_states, 16> m_survivor_states;
std::array<long , 16> m_iq_phase_array;
std::array<double, 21> m_sync_ave;
int m_sql_level;
int m_clk_err_counter;
int m_clk_err_timer;
int m_clk_err;
double m_dev_ave;
double m_freq_error;
int m_sample_cnt;
std::complex<double> m_freq_signal;
bool m_fast_afc_mode;
std::complex<double> m_bit_signal;
bool m_imd_valid;
_internal::imd_calculator m_calc_imd;
double m_sample_freq;
double m_agc_ave;
bool m_afc_on;
double m_fperr_ave {};
double m_fferr_ave {};
std::complex<double> m_z1;
std::complex<double> m_z2;
double m_I0 {}; // 4 stage I/Q delay line variables
double m_I1 {};
double m_Q0 {};
double m_Q1 {};
int m_iq_phz_index {};
mode m_rx_mode { mode::bpsk };
bool m_last_bit_zero {};
uint16_t m_bit_acc {};
std::array<uint8_t, 2048> m_VaricodeDecTbl;
bool m_sq_open {};
int m_squelch_speed { 75 };
double m_q_freq_error {};
int m_on_count {};
int m_off_count {};
int m_pcnt {};
int m_ncnt {};
int m_sq_thresh { 50 };
double m_nlp_k;
};
/* ------------------------------------------------------------------------- */
} /* namespace psk */
} /* namespace ham */
#endif /* DECODER_H_ */

Wyświetl plik

@ -0,0 +1,45 @@
/*
* imd_calculator.hpp
*
* Created on: 24-03-2013
* Author: lucck
*/
/* ------------------------------------------------------------------------- */
#ifndef LIBPSK_IMD_CALCULATOR_HPP_
#define LIBPSK_IMD_CALCULATOR_HPP_
/* ------------------------------------------------------------------------- */
#include <complex>
#include <array>
/* ------------------------------------------------------------------------- */
namespace ham {
namespace psk {
namespace _internal {
/* ------------------------------------------------------------------------- */
class imd_calculator {
//Make object noncopyable
imd_calculator(const imd_calculator&) = delete;
imd_calculator& operator=(const imd_calculator&) = delete;
static auto constexpr NUM_FILTERS = 3;
public:
imd_calculator() {}
void reset();
bool calc_energies( std::complex<double> samp );
bool calc_value( int &imd_val );
private:
std::array<double, NUM_FILTERS> I1 {{}};
std::array<double, NUM_FILTERS> I2 {{}};
std::array<double, NUM_FILTERS> Q1 {{}};
std::array<double, NUM_FILTERS> Q2 {{}};
std::array<double, NUM_FILTERS> m_energy {{}};
int m_ncount {};
double m_snr {};
double m_imd {};
};
/* ------------------------------------------------------------------------- */
} /* namespace _internal */
} /* namespace psk */
} /* namespace ham */
/* ------------------------------------------------------------------------- */
#endif /* IMD_CALCULATOR_HPP_ */
/* ------------------------------------------------------------------------- */

Plik diff jest za duży Load Diff

Wyświetl plik

@ -0,0 +1,84 @@
/*
* imd_calculator.cpp
*
* Created on: 24-03-2013
* Author: lucck
*/
/* ------------------------------------------------------------------------- */
#include "psk/imd_calculator.hpp"
/* ------------------------------------------------------------------------- */
namespace ham {
namespace psk {
namespace _internal {
/* ------------------------------------------------------------------------- */
namespace
{
constexpr auto PI2 = 8.0 * std::atan(1.0);
constexpr auto N = 288; //96 x 2 = Goertzel length(must be an integer value)
constexpr auto FS = 500.0; // sample frequency
constexpr auto F0 = 15.625; // bin frequencies
constexpr auto F1 = 31.25;
constexpr auto F2 = 46.875;
constexpr auto K0 = double(N)*F0/FS;
constexpr auto K1 = double(N)*F1/FS;
constexpr auto K2 = double(N)*F2/FS;
static constexpr double COEF[] =
{
2.0*std::cos(PI2*K0/double(N)),
2.0*std::cos(PI2*K1/double(N)),
2.0*std::cos(PI2*K2/double(N))
};
}
/* ------------------------------------------------------------------------- */
void imd_calculator::reset()
{
I1.fill(0);
I2.fill(0);
Q1.fill(0);
Q2.fill(0);
m_ncount = 0;
}
/* ------------------------------------------------------------------------- */
bool imd_calculator::calc_energies( std::complex<double> samp )
{
std::complex<double> temp;
for(int i=0; i<NUM_FILTERS;i++)
{
temp = std::complex<double>(I1[i], Q1[i]);
I1[i] = I1[i]*COEF[i]-I2[i]+samp.real();
Q1[i] = Q1[i]*COEF[i]-Q2[i]+samp.imag();
I2[i] = temp.real(); Q2[i] = temp.imag();
}
if( ++m_ncount >= N )
{
m_ncount = 0;
for(int i=0; i<NUM_FILTERS;i++)
{
m_energy[i] = I1[i]*I1[i] + I2[i]*I2[i] - I1[i]*I2[i]*COEF[i] +
Q1[i]*Q1[i] + Q2[i]*Q2[i] - Q1[i]*Q2[i]*COEF[i];
I1[i] = I2[i] = Q1[i] = Q2[i] = 0.0;
}
return true;
}
else
return false;
}
/* ------------------------------------------------------------------------- */
bool imd_calculator::calc_value( int &imd_val )
{
m_snr = 10.0*std::log10(m_energy[0]/m_energy[1]);
m_imd = 10.0*std::log10(m_energy[2]/m_energy[0]);
imd_val = int(m_imd);
return m_snr > (-m_imd+6);
}
/* ------------------------------------------------------------------------- */
} /* namespace _internal */
} /* namespace psk */
} /* namespace ham */
/* ------------------------------------------------------------------------- */

106
psk31test.cpp 100644
Wyświetl plik

@ -0,0 +1,106 @@
//http://www.inb.uni-luebeck.de/~boehme/using_libavcodec.html
//gcc --std=gnu99 -o tutorial01 psk31test.c -lavformat -lavcodec -lz -lavutil
#include <stdint.h>
#include <functional>
#include <cstdio>
extern "C" {
#include <libavcodec/avcodec.h>
#include <libavformat/avformat.h>
}
#include <psk/decoder.hpp>
namespace _internal
{
void on_new_samples( int16_t *sample, std::size_t count );
}
int main(int argc, const char * const *argv )
{
if(argc < 2)
{
printf("INvalid argument count\n");
return -1;
}
av_register_all();
AVFormatContext *pFormatCtx = avformat_alloc_context();
// Open video file
if(avformat_open_input(&pFormatCtx, argv[1], NULL, 0)!=0)
{
printf("Couldn't open file\n");
return -1;
}
if(avformat_find_stream_info(pFormatCtx, NULL) <0)
{
printf("Couldn't find stream\n");
return -1;
}
av_dump_format(pFormatCtx, 0, argv[1], 0);
int audioStream = -1;
for( int i=0; i<pFormatCtx->nb_streams; i++)
{
if( pFormatCtx->streams[i]->codec->codec_type==AVMEDIA_TYPE_AUDIO )
{
audioStream = i;
break;
}
}
if(audioStream==-1)
{
printf("Couldn't find audio stream\n");
return -1;
}
AVCodecContext *pCodecCtx = pFormatCtx->streams[audioStream]->codec;
AVCodec *pCodec = avcodec_find_decoder( pCodecCtx->codec_id );
if(pCodec == NULL)
{
printf("Couldn't find codec\n");
return -1;
}
//bitstreams where frame boundaries can fall in the middle of packets
if(pCodec->capabilities & CODEC_CAP_TRUNCATED)
pCodecCtx->flags|=CODEC_FLAG_TRUNCATED;
// Open codec
if(avcodec_open2(pCodecCtx, pCodec, NULL)<0)
{
printf("Couldn't open avcodec\n");
return -1;
}
AVFrame *decoded_frame = avcodec_alloc_frame();
AVPacket *avpkt = reinterpret_cast<AVPacket*>(malloc(sizeof(AVPacket)));
ham::psk::decoder psk_dec( 8000 );
while(av_read_frame(pFormatCtx, avpkt)>=0)
{
int got_frame = 0;
int len = 0;
{
len = avcodec_decode_audio4(pCodecCtx, decoded_frame, &got_frame, avpkt );
if(len < 0)
{
printf("Decode error\n");
return -1;
}
if(got_frame)
{
int data_size = av_samples_get_buffer_size(NULL,pCodecCtx->channels,
decoded_frame->nb_samples, pCodecCtx->sample_fmt, 1);
psk_dec( reinterpret_cast<ham::psk::sample_type*>(decoded_frame->data[0]), data_size );
}
}
if(len < 0)
{
printf("Error when decoding\n");
return -1;
}
}
free( avpkt );
free( decoded_frame );
return 0;
}

150
unix.mk 100644
Wyświetl plik

@ -0,0 +1,150 @@
# Automatic makefile for GNUARM (C/C++)
#Definicje programow
CC ?= $(CROSS_COMPILE)gcc
CXX ?= $(CROSS_COMPILE)c++
AR ?= $(CROSS_COMPILE)ar
CP ?= $(CROSS_COMPILE)objcopy
OBJDUMP ?= $(CROSS_COMPILE)objdump
SIZE ?= $(CROSS_COMPILE)size
#Current platform
PLATFORM_DEVICE ?= BFC1
#Pozostale ustawienia kompilatora
ASFLAGS +=
CFLAGS += -O$(OPT)
CXXFLAGS += -O$(OPT)
CPFLAGS = -S
ARFLAGS = rcs
ifeq ($(LISTING),y)
ASLST = -Wa,-adhlns=$(<:%.S=%.lst)
CLST = -Wa,-adhlns=$(<:%.c=%.lst)
CPPLST = -Wa,-adhlns=$(<:%.cpp=%.lst)
LSSTARGET = $(TARGET).lss
endif
ifeq ($(DEBUG),y)
CFLAGS += -g -DPDEBUG
CXXFLAGS += -g -DPDEBUG
LDFLAGS += -g -DPDEBUG
ASFLAGS += -gstabs -DPDEBUG
else
CFLAGS += -fomit-frame-pointer
CXXFLAGS += -fomit-frame-pointer
LDFLAGS += -fomit-frame-pointer
ASFLAGS += -fomit-frame-pointer
#Remove unused functions
CFLAGS += -ffunction-sections -fdata-sections
CXXFLAGS += -ffunction-sections -fdata-sections
LDFLAGS+= -Wl,--gc-sections
endif
ifeq ($(PLATFORM_DEVICE), BF210)
CFLAGS += -DPLCXML_BF210_PLATFORM
CXXFLAGS += -DPLCXML_BF210_PLATFORM
endif
all: build
install: build program
clean:
rm -f $(TARGET)
rm -f $(TARGET).map
rm -f $(TARGET).lss
rm -f lib$(TARGET).a
rm -f $(OBJ) $(LST) $(DEPFILES) $(LIBS) $(LIBS_OBJS)
ifeq ($(LIBRARY),y)
build: compile
else
build: compile size-calc
endif
ifeq ($(LIBRARY),y)
compile: lib$(TARGET).a
else
compile: $(TARGET) $(LSSTARGET)
endif
#Calculate size
size-calc: $(TARGET)
$(SIZE) $<
#wszystkie zaleznosci
$(TARGET): $(OBJ) $(LSCRIPT)
#Tworzenie biblioteki
lib$(TARGET).a: $(OBJ)
#Depend files
DEPFILES += $(SRC:%.c=%.dep) $(CPPSRC:%.cpp=%.dep) $(ASRC:%.S=%.dep)
-include $(DEPFILES)
#Objects files
OBJ = $(SRC:%.c=%.o) $(CPPSRC:%.cpp=%.o) $(ASRC:%.S=%.o)
# Define all listing files.
LST = $(SRC:%.c=%.lst) $(CPPSRC:%.cpp=%.lst) $(ASRC:%.S=%.lst)
#Objects files
.PRECIOUS : $(OBJ)
ifeq ($(LIBRARY),y)
.SECONDARY: lib$(TARGET).a
else
.SECONDARY: $(TARGET)
endif
.DEFAULT_GOAL := all
%.dep: %.c
$(CC) -MM -MF $@ -MP -MT $(subst .dep,.o,$@) $(CFLAGS) $<
%.dep: %.cpp
$(CXX) -MM -MF $@ -MP -MT $(subst .dep,.o,$@) $(CXXFLAGS) $<
%.dep: %.S
$(CC) -MM -MF $@ -MP -MT $(subst .dep,.o,$@) $(ASFLAGS) $<
%.lss: $(TARGET)
@echo "Create extended listing..."
$(OBJDUMP) -h -S $< > $@
%.hex: %.elf
@echo "Converting to hex..."
$(CP) -O ihex $(CPFLAGS) $< $@
%.bin: %.elf
@echo "Converting to bin..."
$(CP) -O binary $(CPFLAGS) $< $@
$(TARGET): $(OBJ) $(CRT0_OBJECT) $(ADDITIONAL_DEPS) $(LIBS)
@echo "Linking..."
$(CXX) $(CXXFLAGS) $(OBJ) $(CRT0_OBJECT) $(LIBS) -o $@ $(LDFLAGS)
%.o : %.S
@echo "Assembling..."
$(CC) -c $(ASFLAGS) $(ASLST) $< -o $@
%.o : %.c
@echo "Compiling C..."
$(CC) -c $(CFLAGS) $(CLST) $< -o $@
%.o : %.cpp
@echo "Compiling C++..."
$(CXX) -c $(CXXFLAGS) $(CPPLST) $< -o $@
lib$(TARGET).a : $(OBJ)
@echo "Creating library ..."
$(AR) $(ARFLAGS) $@ $(OBJ)