Statistics
| Branch: | Tag: | Revision:

root / gnuradio-core / src / lib / filter / gr_pfb_interpolator_ccf.cc @ a0d13b42

History | View | Annotate | Download (3.9 kB)

1
/* -*- c++ -*- */
2
/*
3
 * Copyright 2009 Free Software Foundation, Inc.
4
 * 
5
 * This file is part of GNU Radio
6
 * 
7
 * GNU Radio is free software; you can redistribute it and/or modify
8
 * it under the terms of the GNU General Public License as published by
9
 * the Free Software Foundation; either version 3, or (at your option)
10
 * any later version.
11
 * 
12
 * GNU Radio is distributed in the hope that it will be useful,
13
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
15
 * GNU General Public License for more details.
16
 * 
17
 * You should have received a copy of the GNU General Public License
18
 * along with GNU Radio; see the file COPYING.  If not, write to
19
 * the Free Software Foundation, Inc., 51 Franklin Street,
20
 * Boston, MA 02110-1301, USA.
21
 */
22
23
#ifdef HAVE_CONFIG_H
24
#include "config.h"
25
#endif
26
27
#include <gr_pfb_interpolator_ccf.h>
28
#include <gr_fir_ccf.h>
29
#include <gr_fir_util.h>
30
#include <gr_io_signature.h>
31
32
gr_pfb_interpolator_ccf_sptr gr_make_pfb_interpolator_ccf (unsigned int interp, 
33
                                                           const std::vector<float> &taps)
34
{
35
  return gr_pfb_interpolator_ccf_sptr (new gr_pfb_interpolator_ccf (interp, taps));
36
}
37
38
39
gr_pfb_interpolator_ccf::gr_pfb_interpolator_ccf (unsigned int interp, 
40
                                                  const std::vector<float> &taps)
41
  : gr_sync_interpolator ("pfb_interpolator_ccf",
42
                          gr_make_io_signature (1, 1, sizeof(gr_complex)),
43
                          gr_make_io_signature (1, 1, sizeof(gr_complex)),
44
                          interp),
45
    d_updated (false)
46
{
47
  d_rate = interp;
48
  d_filters = std::vector<gr_fir_ccf*>(d_rate);
49
50
  // Create an FIR filter for each channel and zero out the taps
51
  std::vector<float> vtaps(0, d_rate);
52
  for(unsigned int i = 0; i < d_rate; i++) {
53
    d_filters[i] = gr_fir_util::create_gr_fir_ccf(vtaps);
54
  }
55
56
  // Now, actually set the filters' taps
57
  set_taps(taps);
58
}
59
60
gr_pfb_interpolator_ccf::~gr_pfb_interpolator_ccf ()
61
{
62
  for(unsigned int i = 0; i < d_rate; i++) {
63
    delete d_filters[i];
64
  }
65
}
66
67
void
68
gr_pfb_interpolator_ccf::set_taps (const std::vector<float> &taps)
69
{
70
  unsigned int i,j;
71
72
  unsigned int ntaps = taps.size();
73
  d_taps_per_filter = (unsigned int)ceil((double)ntaps/(double)d_rate);
74
75
  // Create d_numchan vectors to store each channel's taps
76
  //std::vector< std::vector<float> > vtaps(d_rate);
77
  d_taps.resize(d_rate);
78
79
  // Make a vector of the taps plus fill it out with 0's to fill
80
  // each polyphase filter with exactly d_taps_per_filter
81
  std::vector<float> tmp_taps;
82
  tmp_taps = taps;
83
  while((float)(tmp_taps.size()) < d_rate*d_taps_per_filter) {
84
    tmp_taps.push_back(0.0);
85
  }
86
  
87
  // Partition the filter
88
  for(i = 0; i < d_rate; i++) {
89
    // Each channel uses all d_taps_per_filter with 0's if not enough taps to fill out
90
    d_taps[i] = std::vector<float>(d_taps_per_filter, 0);
91
    for(j = 0; j < d_taps_per_filter; j++) {
92
      d_taps[i][j] = tmp_taps[i + j*d_rate];  // add taps to channels in reverse order
93
    }
94
    
95
    // Build a filter for each channel and add it's taps to it
96
    d_filters[i]->set_taps(d_taps[i]);
97
  }
98
99
  // Set the history to ensure enough input items for each filter
100
  set_history (d_taps_per_filter);
101
102
  d_updated = true;
103
}
104
105
void
106
gr_pfb_interpolator_ccf::print_taps()
107
{
108
  unsigned int i, j;
109
  for(i = 0; i < d_rate; i++) {
110
    printf("filter[%d]: [", i);
111
    for(j = 0; j < d_taps_per_filter; j++) {
112
      printf(" %.4e", d_taps[i][j]);
113
    }
114
    printf("]\n\n");
115
  }
116
}
117
118
int
119
gr_pfb_interpolator_ccf::work (int noutput_items,
120
                               gr_vector_const_void_star &input_items,
121
                               gr_vector_void_star &output_items)
122
{
123
  gr_complex *in = (gr_complex *) input_items[0];
124
  gr_complex *out = (gr_complex *) output_items[0];
125
126
  if (d_updated) {
127
    d_updated = false;
128
    return 0;                     // history requirements may have changed.
129
  }
130
131
  int i = 0, count = 0;
132
133
  while(i < noutput_items) {
134
    for(int j = 0; j < d_rate; j++) {
135
      out[i] = d_filters[j]->filter(&in[count]);
136
      i++;
137
    }
138
    count++;
139
  }
140
  
141
  return i;
142
}