/*
* oscdump - Receive and dump OpenSound Control messages.
*
* Copyright (C) 2008 Kentaro Fukuchi <kentaro@fukuchi.org>
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public License
* as published by the Free Software Foundation; either version 2.1 of
* the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
* 02110-1301 USA.
*
*/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <ctype.h>
#include <errno.h>
#include <signal.h>
#include <config.h>
#include <lo/lo.h>
int done = 0;
int bundled = 0;
lo_timetag tt_now;
lo_timetag tt_bundle;
void usage(void)
{
printf("oscdump version %s\n"
"Copyright (C) 2008 Kentaro Fukuchi\n\n"
"Usage: oscdump [-L] <port>\n"
"or oscdump [-L] <url>\n"
"Receive OpenSound Control messages and dump to standard output.\n\n"
"Description\n"
"-L : specifies line buffering even if stdout is a pipe or file\n"
"port : specifies the listening port number.\n"
"url : specifies the server parameters using a liblo URL.\n"
" e.g. UDP \"osc.udp://:9000\"\n"
" Multicast \"osc.udp://224.0.1.9:9000\"\n"
" TCP \"osc.tcp://:9000\"\n\n", VERSION);
}
int bundleStartHandler(lo_timetag tt, void *user_data)
{
if (tt.sec == LO_TT_IMMEDIATE.sec &&
tt.frac == LO_TT_IMMEDIATE.frac)
{
lo_timetag_now(&tt_now);
tt_bundle.sec = tt_now.sec;
tt_bundle.frac = tt_now.frac;
}
else {
tt_bundle.sec = tt.sec;
tt_bundle.frac = tt.frac;
}
bundled = 1;
return 0;
}
int bundleEndHandler(void *user_data)
{
bundled = 0;
return 0;
}
void errorHandler(int num, const char *msg, const char *where)
{
printf("liblo server error %d in path %s: %s\n", num, where, msg);
}
int messageHandler(const char *path, const char *types, lo_arg ** argv,
int argc, lo_message msg, void *user_data)
{
int i;
if (bundled) {
printf("%08x.%08x %s %s", tt_bundle.sec, tt_bundle.frac, path, types);
}
else {
lo_timetag tt = lo_message_get_timestamp(msg);
if (tt.sec == LO_TT_IMMEDIATE.sec &&
tt.frac == LO_TT_IMMEDIATE.frac)
{
lo_timetag_now(&tt_now);
printf("%08x.%08x %s %s", tt_now.sec, tt_now.frac, path, types);
}
else
printf("%08x.%08x %s %s", tt.sec, tt.frac, path, types);
}
for (i = 0; i < argc; i++) {
printf(" ");
lo_arg_pp((lo_type) types[i], argv[i]);
}
printf("\n");
return 0;
}
void ctrlc(int sig)
{
done = 1;
}
int main(int argc, char **argv)
{
lo_server server;
char *port=0, *group=0;
int i=1;
if (argc > i && argv[i][0]=='-') {
#ifdef HAVE_SETVBUF
if (argv[i][1]=='L') { // line buffering
setvbuf(stdout, 0, _IOLBF, BUFSIZ);
i++;
}
else
#endif
if (argv[i][1]=='h') {
usage();
exit(0);
}
else {
printf("Unknown option `%s'\n", argv[i]);
exit(1);
}
}
if (argc > i) {
port = argv[i];
i++;
} else {
usage();
exit(1);
}
if (argc > i) {
group = argv[i];
}
if (group) {
server = lo_server_new_multicast(group, port, errorHandler);
} else if (isdigit(port[0])) {
server = lo_server_new(port, errorHandler);
} else {
server = lo_server_new_from_url(port, errorHandler);
}
if (server == NULL) {
fprintf(stderr, "Could not start a server with port %s", port);
if (group)
fprintf(stderr, ", multicast group %s\n", group);
else
fprintf(stderr, "\n");
exit(1);
}
lo_server_add_method(server, NULL, NULL, messageHandler, NULL);
lo_server_add_bundle_handlers(server, bundleStartHandler, bundleEndHandler,
NULL);
signal(SIGINT, ctrlc);
while (!done) {
lo_server_recv_noblock(server, 1);
}
return 0;
}