3 /// In the same vein as pppoe, used with pppd to create a
4 /// pty tunnel and GPRS modem link.
8 Copyright (C) 2007-2012, Net Direct Inc. (http://www.netdirect.ca/)
10 This program is free software; you can redistribute it and/or modify
11 it under the terms of the GNU General Public License as published by
12 the Free Software Foundation; either version 2 of the License, or
13 (at your option) any later version.
15 This program is distributed in the hope that it will be useful,
16 but WITHOUT ANY WARRANTY; without even the implied warranty of
17 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
19 See the GNU General Public License in the COPYING file at the
20 root directory of this project for more details.
23 #include <barry/barry.h>
31 #include <sys/select.h>
33 #include <sys/types.h>
38 #include "barrygetopt.h"
41 using namespace Barry
;
43 bool data_dump
= false;
44 volatile bool signal_end
= false;
48 int logical
, major
, minor
;
49 const char *Version
= Barry::Version(logical
, major
, minor
);
52 << "pppob - PPP over Barry\n"
53 << " Copyright 2007-2012, Net Direct Inc. (http://www.netdirect.ca/)\n"
54 << " Using: " << Version
<< "\n"
56 << " -l file Direct pppob log output to file (useful with -v)\n"
57 << " -p pin PIN of device to talk with\n"
58 << " If only one device plugged in, this flag is optional\n"
59 << " -P pass Simplistic method to specify device password\n"
60 << " -s Use Serial mode instead of IpModem\n"
61 << " -v Dump protocol data during operation (debugging only!)\n"
65 void signal_handler(int signum
)
70 void SerialDataCallback(void *context
, const unsigned char *data
, int len
)
72 if( len
&& data_dump
)
73 barryverbose("ReadThread:\n" << Data(data
, len
));
76 int written
= write(1, data
, len
);
82 barryverbose("Error in write()");
87 void ProcessStdin(Modem
&modem
)
89 // Read from stdin and write to USB, until
97 // Handle interrupt signals from pppd
99 signal(SIGINT
, &signal_handler
);
100 signal(SIGHUP
, &signal_handler
);
101 signal(SIGTERM
, &signal_handler
);
104 while( signal_end
== false ) {
105 // Need to use select() here, so that pppd doesn't
106 // hang when it tries to set the line discipline
113 ret
= select(1, &rfds
, NULL
, NULL
, &tv
);
117 else if( ret
&& FD_ISSET(0, &rfds
) ) {
118 bytes_read
= read(0, data
.GetBuffer(), data
.GetBufSize());
119 if( bytes_read
== 0 )
122 if( bytes_read
> 0 ) {
123 data
.ReleaseBuffer(bytes_read
);
130 int main(int argc
, char *argv
[])
134 cout
.sync_with_stdio(true); // leave this on, since libusb uses
135 // stdio for debug messages
140 bool force_serial
= false;
142 std::string password
;
144 // check for options via the fifo first, so the command
145 // line args can override them
147 if( fifo
.Fetch(4) ) {
148 const FifoArgs
&args
= fifo
.GetArgs();
149 pin
= args
.m_pin
.Value();
150 force_serial
= args
.m_use_serial_mode
;
151 logfile
= args
.m_log_filename
;
152 password
= args
.m_password
;
153 data_dump
= args
.m_verbose
;
156 // process command line options
158 int cmd
= getopt(argc
, argv
, "l:p:P:sv");
164 case 'l': // Verbose log file
168 case 'p': // Blackberry PIN
169 pin
= strtoul(optarg
, NULL
, 16);
172 case 'P': // Device password
176 case 's': // Use Serial mode
180 case 'v': // data dump on
191 // Initialize the barry library. Must be called before
193 // Log to stderr, since stdout is for data in this program.
194 std::auto_ptr
<std::ofstream
> log
;
195 if( logfile
.size() ) {
196 log
.reset( new std::ofstream(logfile
.c_str(), ios::app
) );
197 Barry::Init(data_dump
, log
.get());
200 Barry::Init(data_dump
, &std::cerr
);
203 // Display version if in data_dump mode
205 int logical
, major
, minor
;
206 const char *Version
= Barry::Version(logical
, major
, minor
);
207 barryverbose(Version
);
210 // Probe the USB bus for Blackberry devices and display.
211 // If user has specified a PIN, search for it in the
212 // available device list here as well
214 int activeDevice
= probe
.FindActive(pin
);
215 if( activeDevice
== -1 ) {
217 cerr
<< "PIN " << setbase(16) << pin
218 << " not found" << endl
;
219 cerr
<< "No device selected" << endl
;
223 const ProbeResult
&device
= probe
.Get(activeDevice
);
225 if( !force_serial
&& device
.HasIpModem() ) {
226 barryverbose("Using IpModem mode...");
228 // Create our controller object using our threaded router.
229 Controller
con(probe
.Get(activeDevice
));
231 // Open serial mode... the callback handles reading from
232 // USB and writing to stdout
233 Mode::IpModem
modem(con
, SerialDataCallback
, 0);
234 modem
.Open(password
.c_str());
237 modem
.Close(); // graceful close so we can restart without unplugging
241 barryverbose("Using Serial mode per command line...");
244 barryverbose("No IpModem mode available, using Serial mode...");
247 // Create our socket router and start thread to handle
248 // the USB reading, instead of creating our own thread.
249 SocketRoutingQueue router
;
250 router
.SpinoffSimpleReadThread();
252 // Create our controller object using our threaded router.
253 Controller
con(probe
.Get(activeDevice
), router
);
255 // Open desktop mode... this handles the password side
257 Mode::Desktop
desktop(con
);
258 desktop
.Open(password
.c_str());
260 // Open serial connection
261 Mode::Serial
modem(con
, SerialDataCallback
, 0);
262 modem
.Open(password
.c_str());
267 barryverbose("Exiting");
270 catch( std::exception
&e
) {
271 cerr
<< "exception caught in main(): " << e
.what() << endl
;