blob: 29deee99033cfd8fcb3cf42366597c5b16dd18ed [file] [log] [blame]
#include "vision/BinaryServer.h"
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <sys/mman.h>
#include <errno.h>
#include <string.h>
#include <vector>
#include <netinet/in.h>
#include <netinet/tcp.h>
#include <arpa/inet.h>
#include "aos/externals/libjpeg/include/jpeglib.h"
#include "aos/atom_code/camera/Buffers.h"
#include "aos/common/time.h"
namespace frc971 {
namespace vision {
static void echo_read_cb(struct bufferevent *bev, void * /*ctx*/){
struct evbuffer *input = bufferevent_get_input(bev);
struct evbuffer *output = bufferevent_get_output(bev);
size_t len = evbuffer_get_length(input);
char *data;
data = (char *)malloc(len);
evbuffer_copyout(input, data, len);
printf("we got some data: %s\n", data);
evbuffer_add_buffer(output, input);
}
void BinaryServer::ErrorEvent(struct bufferevent *bev, short events){
if (events & BEV_EVENT_ERROR)
perror("Error from bufferevent");
if (events & (BEV_EVENT_EOF | BEV_EVENT_ERROR)) {
have_id = false;
bufferevent_free(bev);
}
}
void BinaryServer::Accept(struct evconnlistener *listener,
evutil_socket_t fd, struct sockaddr * /*address*/, int /*socklen*/){
struct event_base *base = evconnlistener_get_base(listener);
if(!have_id){
struct bufferevent *bev = bufferevent_socket_new(base, fd, BEV_OPT_CLOSE_ON_FREE);
_output = bufferevent_get_output(bev);
_bufev = bev;
have_id = true;
int no_delay_flag = 1;
setsockopt(fd, IPPROTO_TCP, TCP_NODELAY, &no_delay_flag, sizeof(no_delay_flag));
bufferevent_setcb(bev, echo_read_cb, NULL, StaticErrorEvent, this);
bufferevent_enable(bev, EV_READ | EV_WRITE);
}
}
static void accept_error_cb(struct evconnlistener *listener, void * /*ctx*/)
{
struct event_base *base = evconnlistener_get_base(listener);
int err = EVUTIL_SOCKET_ERROR();
fprintf(stderr, "Got an error %d (%s) on the listener. "
"Shutting down.\n", err, evutil_socket_error_to_string(err));
event_base_loopexit(base, NULL);
}
void BinaryServer::StartServer(uint16_t port){
_fd = socket(AF_INET, SOCK_STREAM, 0);
struct sockaddr_in sin;
memset(&sin, 0, sizeof(sin));
sin.sin_family = AF_INET;
sin.sin_port = htons( port );
sin.sin_addr.s_addr = inet_addr("0.0.0.0");
listener = evconnlistener_new_bind(_base, StaticAccept, this,
LEV_OPT_CLOSE_ON_FREE | LEV_OPT_REUSEABLE, -1,
(struct sockaddr *) (void *)&sin, sizeof(sin));
if (!listener) {
fprintf(stderr,"%s:%d: Couldn't create listener\n", __FILE__, __LINE__);
exit(-1);
}
evconnlistener_set_error_cb(listener, accept_error_cb);
}
void BinaryServer::Notify(int fd,short /*what*/){
char notes[4096];
int count = read(fd,notes,4096);
if(count == 0){
close(fd);
fprintf(stderr,"%s:%d: Error No cheeze from OpenCV task!!!\n",__FILE__,__LINE__);
exit(-1);
}
printf("notified!: %d\n",count);
if(have_id){
printf("got someone to read my stuff!\n");
char *binary_data;
size_t len;
if(_notify->GetData(&binary_data,&len)){
printf("here is for sending\n");
evbuffer_add_reference(_output, binary_data, len, PacketNotifier::StaticDataSent,_notify);
printf("this is how sending went %d\n",bufferevent_flush(_bufev,EV_WRITE, BEV_FLUSH));
}
}
}
//Constructor
BinaryServer::BinaryServer(uint16_t port,
frc971::vision::PacketNotifier *notify) :
_base(event_base_new()) {
have_id = false;
StartServer(port);
_notify = notify;
frame_notify = event_new(_base, notify->RecieverFD(),
EV_READ|EV_PERSIST, StaticNotify, this);
event_add(frame_notify,NULL);
event_base_dispatch(_base);
}
} // namespace vision
} // namespace frc971