3b01f224b102daaff8c6586b6671eafd25c869c2
[senf.git] / Socket / Protocols / DVB / DVBFrontendHandle.cc
1 // $Id$
2 //
3 // Copyright (C) 2007
4 // Fraunhofer Institute for Open Communication Systems (FOKUS)
5 // Competence Center NETwork research (NET), St. Augustin, GERMANY
6 //     Thorsten Horstmann <tho@berlios.de>
7 //
8 // This program is free software; you can redistribute it and/or modify
9 // it under the terms of the GNU General Public License as published by
10 // the Free Software Foundation; either version 2 of the License, or
11 // (at your option) any later version.
12 //
13 // This program is distributed in the hope that it will be useful,
14 // but WITHOUT ANY WARRANTY; without even the implied warranty of
15 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16 // GNU General Public License for more details.
17 //
18 // You should have received a copy of the GNU General Public License
19 // along with this program; if not, write to the
20 // Free Software Foundation, Inc.,
21 // 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
22
23 /** \file
24     \brief DVBFrontendHandle non-inline non-template implementation */
25
26 #include "DVBFrontendHandle.hh"
27 //#include "DVBFrontendHandle.ih"
28
29 // Custom includes
30 #include <boost/format.hpp>
31 #include <sys/socket.h>
32 #include "../../../Utils/Exception.hh"
33 #include <sys/ioctl.h>
34
35
36 //#include "DVBFrontendHandle.mpp"
37 #define prefix_
38 ///////////////////////////////cc.p////////////////////////////////////////
39
40 ///////////////////////////////////////////////////////////////////////////
41 // senf::DVBFrontendHandle
42 using namespace std;
43
44 prefix_ void senf::DVBFrontendSocketProtocol::init_client(unsigned short adapter, unsigned short device, int flags)
45     const
46 {
47     string devFrontend = str( boost::format("/dev/dvb/adapter%d/frontend%d") % adapter % device);
48     int f = open(devFrontend.c_str(), flags);
49     if (f < 0)
50         SENF_THROW_SYSTEM_EXCEPTION("")<< "Could not open frontend device of DVB adapter " << devFrontend << ".";
51     fd(f);
52 }
53
54 prefix_ unsigned senf::DVBFrontendSocketProtocol::available()
55     const
56 {
57     return 0;
58 }
59
60 prefix_ bool senf::DVBFrontendSocketProtocol::eof()
61     const
62 {
63     return false;
64 }
65
66 prefix_ void senf::DVBFrontendSocketProtocol::tune(const struct dvb_frontend_parameters & frontend)
67     const 
68 {
69     // tuning
70     if (::ioctl(fd(), FE_SET_FRONTEND, &frontend) )
71         SENF_THROW_SYSTEM_EXCEPTION("") << "ioctl FE_SET_FRONTEND failed. Socket should initialized with r/w permissions.";;
72 }
73
74 prefix_ void senf::DVBFrontendSocketProtocol::tuneDVB_T(unsigned int frequency, 
75         fe_spectral_inversion_t inversion,
76         fe_bandwidth_t bandwidth, 
77         fe_code_rate_t code_rate_HP, /* high priority stream code rate */
78         fe_code_rate_t code_rate_LP, /* low priority stream code rate */
79         fe_modulation_t constellation, /* modulation type */
80         fe_transmit_mode_t transmission_mode, 
81         fe_guard_interval_t guard_interval,
82         fe_hierarchy_t hierarchy_information
83         )
84     const
85 {
86     struct dvb_ofdm_parameters ofdm; /* DVB-T Parameters */
87     struct dvb_frontend_parameters frontend;
88     
89     ::memset(&frontend, 0, sizeof(struct dvb_frontend_parameters));
90     ::memset(&ofdm, 0, sizeof(struct dvb_ofdm_parameters));
91     
92     ofdm.bandwidth = bandwidth;
93     ofdm.code_rate_HP = code_rate_HP;
94     ofdm.code_rate_LP = code_rate_LP;
95     ofdm.constellation = constellation; 
96     ofdm.guard_interval = guard_interval;
97     ofdm.hierarchy_information = hierarchy_information;
98     
99     frontend.frequency = frequency;
100     frontend.inversion = inversion;
101     frontend.u.ofdm = ofdm;
102     
103     tune(frontend);
104     
105 }
106 prefix_ void senf::DVBFrontendSocketProtocol::tuneDVB_S(unsigned int frequency, 
107         fe_spectral_inversion_t inversion, 
108         unsigned int symbole_rate, /* symbol rate in Symbols per second */
109         fe_code_rate_t fec_inner /* forward error correction (see above) */
110     )
111     const
112 {
113     struct dvb_qpsk_parameters qpsk; /* DVB-S Parameters*/
114     struct dvb_frontend_parameters frontend;
115     
116     ::memset(&frontend, 0, sizeof(struct dvb_frontend_parameters));
117     ::memset(&qpsk, 0, sizeof(struct dvb_qpsk_parameters));
118     
119     qpsk.symbol_rate = symbole_rate;
120     qpsk.fec_inner = fec_inner;
121     
122     frontend.frequency = frequency;
123     frontend.inversion = inversion;
124     frontend.u.qpsk = qpsk;
125     
126     tune(frontend);
127 }
128
129 prefix_ void senf::DVBFrontendSocketProtocol::tuneDVB_C(unsigned int frequency, 
130         fe_spectral_inversion_t inversion,
131         unsigned int symbol_rate,
132         fe_code_rate_t fec_inner,
133         fe_modulation_t modulation
134         )
135 const
136 {
137     struct dvb_qam_parameters qam; /* DVB-C Parameters*/
138     struct dvb_frontend_parameters frontend;
139     
140     ::memset(&frontend, 0, sizeof(struct dvb_frontend_parameters));
141     ::memset(&qam, 0, sizeof(struct dvb_qam_parameters));
142     
143     qam.symbol_rate = symbol_rate;
144     qam.fec_inner = fec_inner;
145     qam.modulation = modulation;
146     
147     
148     frontend.frequency = frequency;
149     frontend.inversion = inversion;
150     frontend.u.qam = qam;
151        
152     tune(frontend);
153 }
154 prefix_ void senf::DVBFrontendSocketProtocol::setNonBlock(bool on) const{
155    if(on)
156        ::fcntl(fd(), F_SETFL, ::fcntl(fd(), F_GETFL) | O_NONBLOCK);
157    else
158        ::fcntl(fd(), F_SETFL, ::fcntl(fd(), F_GETFL) & ~O_NONBLOCK);
159
160 }
161 prefix_ dvb_frontend_info senf::DVBFrontendSocketProtocol::getInfo()
162     const
163 {
164     struct dvb_frontend_info info;
165     ::memset(&info, 0, sizeof(struct dvb_frontend_info));
166     
167     if( ::ioctl(fd(), FE_GET_INFO, &info) ) {
168         SENF_THROW_SYSTEM_EXCEPTION("") << "Could not read on fildescriptor.";
169     }
170     return info;
171 }
172
173 prefix_ dvb_frontend_parameters senf::DVBFrontendSocketProtocol::getFrontendParam() const {
174     struct dvb_frontend_parameters frontend_;
175     
176     ::memset(&frontend_, 0, sizeof(struct dvb_frontend_parameters));
177     
178     if(::ioctl(fd(), FE_GET_FRONTEND, &frontend_)) {
179         switch(errno) {
180             case EBADF:
181                 SENF_THROW_SYSTEM_EXCEPTION("fd is not a valid open file descriptor.");
182                 break;
183             case EFAULT:
184                 SENF_THROW_SYSTEM_EXCEPTION( "frontend_ points to invalid address." );
185                 break;
186             case EINVAL:
187                 SENF_THROW_SYSTEM_EXCEPTION( "Maximum supported symbol rate reached. This call may not be implemented by kernel driver!" );
188                 break;
189             default:
190                 SENF_THROW_SYSTEM_EXCEPTION("Errno: ") << errno;
191         }
192     }
193     return frontend_;
194 }
195
196 prefix_ dvb_frontend_event senf::DVBFrontendSocketProtocol::getEvent() const{
197     struct dvb_frontend_event ev ;
198     
199     ::memset(&ev, 0, sizeof(struct dvb_frontend_event));
200     
201     if(::ioctl(fd(), FE_GET_EVENT, &ev)) {
202         switch(errno) {
203             case EBADF:
204                 SENF_THROW_SYSTEM_EXCEPTION("Could not read from frontend device, read-only access to the device is sufficient.");
205                 break;
206             case EWOULDBLOCK:
207                 SENF_THROW_SYSTEM_EXCEPTION( "No event pending and device is in nonblocking mode." );
208                 break;
209             default:
210                 SENF_THROW_SYSTEM_EXCEPTION("Errno: ") << errno;
211         }
212     }
213     return ev;
214 }
215
216 prefix_ int16_t senf::DVBFrontendSocketProtocol::signalStrength()
217     const
218 {
219     int16_t strength;
220     if (::ioctl(fd(), FE_READ_SIGNAL_STRENGTH, &strength) < 0)
221         SENF_THROW_SYSTEM_EXCEPTION("Could not get signal strength of DVB adapter.");
222     return strength;
223 }
224
225 prefix_ int16_t senf::DVBFrontendSocketProtocol::signalNoiseRatio()
226     const
227 {
228     int16_t snr;
229     if (::ioctl(fd(), FE_READ_SNR, &snr) < 0)
230         SENF_THROW_SYSTEM_EXCEPTION("Could not get signal-to-noise ratio of DVB adapter.");
231     return snr;
232 }
233
234 prefix_ uint32_t senf::DVBFrontendSocketProtocol::bitErrorRate()
235     const
236 {
237     uint32_t ber;
238     if (::ioctl(fd(), FE_READ_BER, &ber) < 0)
239         SENF_THROW_SYSTEM_EXCEPTION("Could not get bit error rate of DVB adapter.");
240     return ber;
241 }
242
243 prefix_ uint32_t senf::DVBFrontendSocketProtocol::uncorrectedBlocks()
244     const
245 {
246     uint32_t uncorrected_blocks;
247     if (::ioctl(fd(), FE_READ_UNCORRECTED_BLOCKS, &uncorrected_blocks) < 0)
248         SENF_THROW_SYSTEM_EXCEPTION("Could not get bit error rate of DVB adapter.");
249     return uncorrected_blocks;
250 }
251
252
253 prefix_ fe_status_t senf::DVBFrontendSocketProtocol::status()
254     const
255 {
256     fe_status_t status;
257     if (::ioctl(fd(), FE_READ_STATUS, &status) < 0)
258         SENF_THROW_SYSTEM_EXCEPTION("Could not get bit error rate of DVB adapter.");
259     return status;
260 }
261
262
263 ///////////////////////////////cc.e////////////////////////////////////////
264 #undef prefix_
265 //#include "DVBFrontendHandle.mpp"
266
267
268 // Local Variables:
269 // mode: c++
270 // fill-column: 100
271 // c-file-style: "senf"
272 // indent-tabs-mode: nil
273 // ispell-local-dictionary: "american"
274 // compile-command: "scons -u test"
275 // comment-column: 40
276 // End: