This program simulates Serially Concatenated Convolutional Codes (SCCCs) of coding rate 1/4 using a turbo decoder with a SISO NSC module and a SISO RSC module.
Reference: S. Benedetto, D. Divsalar, G. Motorsi and F. Pollara, "A Soft-Input Soft-Output Maximum A posteriori (MAP) Module to Decode Parallel and Serial Concatenated Codes", TDA Progress Report, nov. 1996
using namespace itpp;
using std::cout;
using std::endl;
using std::string;
int main(void)
{
double threshold_value = 50;
string map_metric = "maxlogMAP";
ivec gen = "07 05";
int constraint_length = 3;
int nb_errors_lim = 1500;
int nb_bits_lim = int(1e6);
int perm_len =
pow2i(14);
int nb_iter = 10;
vec EbN0_dB = "0:0.1:5";
double R = 1.0 / 4.0;
double Ec = 1.0;
int nb_bits_tail = perm_len / gen.length();
int nb_bits = nb_bits_tail - (constraint_length - 1);
vec sigma2 = (0.5 * Ec / R) *
pow(
inv_dB(EbN0_dB), -1.0);
double Lc;
int nb_blocks;
int nb_errors;
bvec bits(nb_bits);
bvec nsc_coded_bits;
ivec perm(perm_len);
ivec inv_perm(perm_len);
int rec_len = gen.length() * perm_len;
bvec coded_bits(rec_len);
vec rec(rec_len);
vec rsc_intrinsic_coded(rec_len);
vec rsc_apriori_data(perm_len);
vec rsc_extrinsic_coded;
vec rsc_extrinsic_data;
vec nsc_intrinsic_coded(perm_len);
vec nsc_apriori_data(nb_bits_tail);
nsc_apriori_data.zeros();
vec nsc_extrinsic_coded;
vec nsc_extrinsic_data;
bvec rec_bits(nb_bits_tail);
int snr_len = EbN0_dB.length();
mat ber(nb_iter, snr_len);
ber.zeros();
register int en, n;
for (en = 0;en < snr_len;en++) {
cout << "EbN0_dB = " << EbN0_dB(en) << endl;
Lc = -2.0 / sigma2(en);
nb_errors = 0;
nb_blocks = 0;
while ((nb_errors < nb_errors_lim) && (nb_blocks*nb_bits < nb_bits_lim))
{
perm = sort_index(
randu(perm_len));
inv_perm = sort_index(perm);
nsc_coded_bits = nsc_coded_bits(perm);
rsc.
encode(nsc_coded_bits, rsc_parity_bits);
for (n = 0;n < perm_len;n++)
{
coded_bits(2*n) = nsc_coded_bits(n);
coded_bits(2*n + 1) = rsc_parity_bits(n, 0);
}
rsc_intrinsic_coded = Lc * rec;
rsc_apriori_data.zeros();
for (n = 0;n < nb_iter;n++)
{
siso.
rsc(rsc_extrinsic_coded, rsc_extrinsic_data, rsc_intrinsic_coded, rsc_apriori_data,
false);
nsc_intrinsic_coded =
SISO::threshold(rsc_extrinsic_data(inv_perm), threshold_value);
siso.
nsc(nsc_extrinsic_coded, nsc_extrinsic_data, nsc_intrinsic_coded, nsc_apriori_data,
true);
berc.
count(bits, rec_bits.left(nb_bits));
rsc_apriori_data = nsc_extrinsic_coded(perm);
}
nb_blocks++;
}
ber.set_col(en, ber.get_col(en) / nb_blocks);
}
ff <<
Name(
"BER") << ber;
ff <<
Name(
"EbN0_dB") << EbN0_dB;
return 0;
}
When you run this program, the results (BER and EbN0_dB) are saved into sccc_bersim_awgn.it file. Using the following MATLAB script
the results can be displayed.