55c48d0fdc49b2ba4396e603e71b859594d65c67
[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 "senf/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) 
155     const
156 {
157    if(on)
158        ::fcntl(fd(), F_SETFL, ::fcntl(fd(), F_GETFL) | O_NONBLOCK);
159    else
160        ::fcntl(fd(), F_SETFL, ::fcntl(fd(), F_GETFL) & ~O_NONBLOCK);
161
162 }
163 prefix_ dvb_frontend_info senf::DVBFrontendSocketProtocol::getInfo()
164     const
165 {
166     struct dvb_frontend_info info;
167     ::memset(&info, 0, sizeof(struct dvb_frontend_info));
168     
169     if( ::ioctl(fd(), FE_GET_INFO, &info) ) {
170         SENF_THROW_SYSTEM_EXCEPTION("") << "Could not read on fildescriptor.";
171     }
172     return info;
173 }
174
175 prefix_ dvb_frontend_parameters senf::DVBFrontendSocketProtocol::getFrontendParam() const {
176     struct dvb_frontend_parameters frontend_;
177     
178     ::memset(&frontend_, 0, sizeof(struct dvb_frontend_parameters));
179     
180     if(::ioctl(fd(), FE_GET_FRONTEND, &frontend_)) {
181         switch(errno) {
182             case EBADF:
183                 SENF_THROW_SYSTEM_EXCEPTION("fd is not a valid open file descriptor.");
184                 break;
185             case EFAULT:
186                 SENF_THROW_SYSTEM_EXCEPTION( "frontend_ points to invalid address." );
187                 break;
188             case EINVAL:
189                 SENF_THROW_SYSTEM_EXCEPTION( "Maximum supported symbol rate reached. This call may not be implemented by kernel driver!" );
190                 break;
191             default:
192                 SENF_THROW_SYSTEM_EXCEPTION("Errno: ") << errno;
193         }
194     }
195     return frontend_;
196 }
197
198 prefix_ dvb_frontend_event senf::DVBFrontendSocketProtocol::getEvent() const{
199     struct dvb_frontend_event ev ;
200     
201     ::memset(&ev, 0, sizeof(struct dvb_frontend_event));
202     
203     if(::ioctl(fd(), FE_GET_EVENT, &ev)) {
204         switch(errno) {
205             case EBADF:
206                 SENF_THROW_SYSTEM_EXCEPTION("Could not read from frontend device, read-only access to the device is sufficient.");
207                 break;
208             case EWOULDBLOCK:
209                 SENF_THROW_SYSTEM_EXCEPTION( "No event pending and device is in nonblocking mode." );
210                 break;
211             default:
212                 SENF_THROW_SYSTEM_EXCEPTION("Errno: ") << errno;
213         }
214     }
215     return ev;
216 }
217
218 prefix_ int16_t senf::DVBFrontendSocketProtocol::signalStrength()
219     const
220 {
221     int16_t strength;
222     if (::ioctl(fd(), FE_READ_SIGNAL_STRENGTH, &strength) < 0)
223         SENF_THROW_SYSTEM_EXCEPTION("Could not get signal strength of DVB adapter.");
224     return strength;
225 }
226
227 prefix_ int16_t senf::DVBFrontendSocketProtocol::signalNoiseRatio()
228     const 
229 {
230     int16_t snr;
231     if (::ioctl(fd(), FE_READ_SNR, &snr) < 0)
232         SENF_THROW_SYSTEM_EXCEPTION("Could not get signal-to-noise ratio of DVB adapter.");
233     return snr;
234 }
235
236 prefix_ uint32_t senf::DVBFrontendSocketProtocol::bitErrorRate()
237     const
238 {
239     uint32_t ber;
240     if (::ioctl(fd(), FE_READ_BER, &ber) < 0)
241         SENF_THROW_SYSTEM_EXCEPTION("Could not get bit error rate of DVB adapter.");
242     return ber;
243 }
244
245 prefix_ uint32_t senf::DVBFrontendSocketProtocol::uncorrectedBlocks()
246     const
247 {
248     uint32_t uncorrected_blocks;
249     if (::ioctl(fd(), FE_READ_UNCORRECTED_BLOCKS, &uncorrected_blocks) < 0)
250         SENF_THROW_SYSTEM_EXCEPTION("Could not get bit error rate of DVB adapter.");
251     return uncorrected_blocks;
252 }
253
254
255 prefix_ fe_status_t senf::DVBFrontendSocketProtocol::status()
256     const
257 {
258     fe_status_t status;
259     if (::ioctl(fd(), FE_READ_STATUS, &status) < 0)
260         SENF_THROW_SYSTEM_EXCEPTION("Could not get bit error rate of DVB adapter.");
261     return status;
262 }
263
264
265 ///////////////////////////////cc.e////////////////////////////////////////
266 #undef prefix_
267 //#include "DVBFrontendHandle.mpp"
268
269
270 // Local Variables:
271 // mode: c++
272 // fill-column: 100
273 // c-file-style: "senf"
274 // indent-tabs-mode: nil
275 // ispell-local-dictionary: "american"
276 // compile-command: "scons -u test"
277 // comment-column: 40
278 // End: