193 lines
6.3 KiB
C
193 lines
6.3 KiB
C
/*
|
|
$id:$
|
|
Copyright (C) 2003 Antoine Drouin
|
|
|
|
This file is part of paparazzi.
|
|
|
|
paparazzi is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; either version 2, or (at your option)
|
|
any later version.
|
|
|
|
paparazzi 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 General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with paparazzi; see the file COPYING. If not, write to
|
|
the Free Software Foundation, 59 Temple Place - Suite 330,
|
|
Boston, MA 02111-1307, USA.
|
|
*/
|
|
|
|
/*
|
|
network functions
|
|
*/
|
|
|
|
#include "network.h"
|
|
|
|
#include <string.h>
|
|
#include <errno.h>
|
|
|
|
#include <netdb.h>
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/socket.h>
|
|
#include <arpa/inet.h>
|
|
|
|
static gboolean server_on_data_received( GIOChannel *source,
|
|
GIOCondition condition, gpointer data );
|
|
static gboolean client_on_data_received( GIOChannel *source,
|
|
GIOCondition condition, gpointer data );
|
|
|
|
struct NetworkServer *network_server_new( struct PprzProtocol *protocol,
|
|
const char *bcast_addr, const int bcast_port,
|
|
GFunc rcv_callback, GFunc err_callback, gpointer user_data )
|
|
{
|
|
struct NetworkServer *this = g_new( struct NetworkServer, 1 );
|
|
if ( this ) {
|
|
GIOChannel *channel;
|
|
struct protoent *pte = getprotobyname( "UDP" );
|
|
static int so_broadcast = TRUE;
|
|
|
|
this->bcast_socket = socket( PF_INET, SOCK_DGRAM, pte->p_proto );
|
|
setsockopt( this->bcast_socket, SOL_SOCKET, SO_BROADCAST, &so_broadcast,
|
|
sizeof( so_broadcast ) );
|
|
|
|
this->bcast_addr.sin_family = PF_INET;
|
|
this->bcast_addr.sin_port = htons( bcast_port );
|
|
this->bcast_addr.sin_addr.s_addr = inet_addr( bcast_addr );
|
|
|
|
channel = g_io_channel_unix_new( this->bcast_socket );
|
|
g_io_add_watch ( channel, G_IO_IN | G_IO_PRI, server_on_data_received, this );
|
|
|
|
this->protocol = protocol;
|
|
|
|
this->rcv_callback = rcv_callback;
|
|
this->err_callback = err_callback;
|
|
this->user_data = user_data;
|
|
}
|
|
return this;
|
|
}
|
|
|
|
#define FIXME_LEN 2048
|
|
|
|
void network_server_dispatch( struct NetworkServer *this, struct PprzMsg *msg )
|
|
{
|
|
guchar buf[ FIXME_LEN ];
|
|
GTimeVal *date = ( GTimeVal * )buf;
|
|
guchar *type = buf + sizeof( GTimeVal );
|
|
guchar *data = type + 1;
|
|
guint len = sizeof( GTimeVal ) + msg->class->size;
|
|
*date = msg->date;
|
|
*type = msg->class->id;
|
|
memcpy( data, msg->bytes, msg->class->size - 1 );
|
|
if ( sendto( this->bcast_socket, buf, len, 0,
|
|
( struct sockaddr * )&this->bcast_addr, sizeof( this->bcast_addr ) ) == -1 )
|
|
g_message( "error broadcasting" );
|
|
}
|
|
|
|
struct NetworkClient *network_client_new( struct PprzProtocol *protocol,
|
|
const char *bcast_addr, const int bcast_port,
|
|
GFunc rcv_callback, GFunc err_callback, gpointer user_data )
|
|
{
|
|
struct NetworkClient *this = g_new( struct NetworkClient, 1 );
|
|
if ( this ) {
|
|
int so_reuseaddr = TRUE;
|
|
GIOChannel *channel;
|
|
struct protoent *pte = getprotobyname( "UDP" );
|
|
|
|
this->socket = socket( PF_INET, SOCK_DGRAM, pte->p_proto );
|
|
setsockopt( this->socket, SOL_SOCKET, SO_REUSEADDR, &so_reuseaddr,
|
|
sizeof( so_reuseaddr ) );
|
|
this->bcast_addr.sin_family = PF_INET;
|
|
this->bcast_addr.sin_port = htons( bcast_port );
|
|
this->bcast_addr.sin_addr.s_addr = inet_addr( bcast_addr );
|
|
bind( this->socket, ( struct sockaddr * )&this->bcast_addr,
|
|
sizeof( this->bcast_addr ) );
|
|
|
|
channel = g_io_channel_unix_new( this->socket );
|
|
g_io_add_watch ( channel, G_IO_IN | G_IO_PRI, client_on_data_received, this );
|
|
|
|
this->protocol = protocol;
|
|
|
|
this->rcv_callback = rcv_callback;
|
|
this->err_callback = err_callback;
|
|
this->user_data = user_data;
|
|
}
|
|
return this;
|
|
}
|
|
|
|
void network_client_send_to_server( struct NetworkClient *this,
|
|
struct PprzMsg *msg )
|
|
{
|
|
guchar buf[ FIXME_LEN ];
|
|
GTimeVal *date = ( GTimeVal * )buf;
|
|
guchar *type = buf + sizeof( GTimeVal );
|
|
guchar *data = type + 1;
|
|
guint len = sizeof( GTimeVal ) + msg->class->size;
|
|
*date = msg->date;
|
|
*type = msg->class->id;
|
|
memcpy( data, msg->bytes, msg->class->size - 1 );
|
|
if ( sendto( this->socket, buf, len, 0,
|
|
( struct sockaddr * )&this->bcast_addr, sizeof( this->bcast_addr ) ) == -1 )
|
|
g_message( "error sending to server" );
|
|
}
|
|
|
|
static gboolean server_on_data_received( GIOChannel *source,
|
|
GIOCondition condition, gpointer data )
|
|
{
|
|
guchar buf[ FIXME_LEN ];
|
|
struct sockaddr_in from;
|
|
int from_len;
|
|
gint len;
|
|
struct NetworkServer *this = ( struct NetworkServer * )data;
|
|
if ( ( len = recvfrom( this->bcast_socket, buf, FIXME_LEN, 0,
|
|
( struct sockaddr * )&from, &from_len ) ) == -1 ) {
|
|
this->err_callback( NULL, this->user_data );
|
|
return FALSE;
|
|
} else {
|
|
struct PprzMsg *msg;
|
|
GTimeVal *date = ( GTimeVal * )buf;
|
|
guchar *ptr_type = buf + sizeof( GTimeVal );
|
|
if ( !( msg = pprz_protocol_msg_new_of_bin( this->protocol, ptr_type,
|
|
*date ) ) ) {
|
|
this->err_callback( NULL, this->user_data );
|
|
return TRUE;
|
|
}
|
|
this->rcv_callback( msg, this->user_data );
|
|
// g_message("network data received %s %d", inet_ntoa(from.sin_addr), ntohs(from.sin_port));
|
|
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean client_on_data_received( GIOChannel *source,
|
|
GIOCondition condition, gpointer data )
|
|
{
|
|
guchar buf[ FIXME_LEN ];
|
|
struct sockaddr_in from;
|
|
int from_len;
|
|
gint len;
|
|
struct NetworkClient *this = ( struct NetworkClient * )data;
|
|
if ( ( len = recvfrom( this->socket, buf, FIXME_LEN, 0,
|
|
( struct sockaddr * )&from, &from_len ) ) == -1 ) {
|
|
g_message( "client_on_data_received [ %s ]", strerror( errno ) );
|
|
this->err_callback( NULL, this->user_data );
|
|
return TRUE;
|
|
} else {
|
|
struct PprzMsg *msg;
|
|
GTimeVal *date = ( GTimeVal * )buf;
|
|
guchar *ptr_type = buf + sizeof( GTimeVal );
|
|
if ( !( msg = pprz_protocol_msg_new_of_bin( this->protocol, ptr_type,
|
|
*date ) ) ) {
|
|
this->err_callback( NULL, this->user_data );
|
|
return TRUE;
|
|
}
|
|
this->rcv_callback( msg, this->user_data );
|
|
// g_message("network data received %s %d", inet_ntoa(from.sin_addr), ntohs(from.sin_port));
|
|
}
|
|
this->server_addr = from;
|
|
return TRUE;
|
|
}
|