/* * Copyright (C) 2004 Steve Harris * * 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. * * $Id$ */ #include #include #include #ifdef WIN32 /* Define to 1 if you have the timespec struct. pthread is going to import time.h and then complain that it wants to redefine something already found in time.h. */ #define HAVE_STRUCT_TIMESPEC 1 #endif #define HAVE_PTHREAD_H 1 #include #include #ifdef WIN32 #include #include #else #include #include #include #endif #include "lo_types_internal.h" #include "lo.h" #include "lo_throw.h" static void thread_func(void *data); lo_server_thread lo_server_thread_new(const char *port, lo_err_handler err_h) { return lo_server_thread_new_with_proto(port, LO_DEFAULT, err_h); } lo_server_thread lo_server_thread_new_multicast(const char *group, const char *port, lo_err_handler err_h) { lo_server_thread st = malloc(sizeof(struct _lo_server_thread)); st->s = lo_server_new_multicast(group, port, err_h); st->active = 0; st->done = 0; if (!st->s) { free(st); return NULL; } return st; } lo_server_thread lo_server_thread_new_with_proto(const char *port, int proto, lo_err_handler err_h) { lo_server_thread st = malloc(sizeof(struct _lo_server_thread)); st->s = lo_server_new_with_proto(port, proto, err_h); st->active = 0; st->done = 0; if (!st->s) { free(st); return NULL; } return st; } void lo_server_thread_free(lo_server_thread st) { if (st) { if (st->active) { lo_server_thread_stop(st); } lo_server_free(st->s); } free(st); } lo_method lo_server_thread_add_method(lo_server_thread st, const char *path, const char *typespec, lo_method_handler h, void *user_data) { return lo_server_add_method(st->s, path, typespec, h, user_data); } void lo_server_thread_del_method(lo_server_thread st, const char *path, const char *typespec) { lo_server_del_method(st->s, path, typespec); } int lo_server_thread_start(lo_server_thread st) { int result; if (!st->active) { st->active = 1; st->done = 0; // Create the server thread result = pthread_create(&(st->thread), NULL, (void *)&thread_func, st); if (result) { fprintf(stderr, "Failed to create thread: pthread_create(), %s", strerror(result)); return -result; } } return 0; } int lo_server_thread_stop(lo_server_thread st) { int result; if (st->active) { // Signal thread to stop st->active = 0; // pthread_join waits for thread to terminate // and then releases the thread's resources result = pthread_join( st->thread, NULL ); if (result) { fprintf(stderr, "Failed to stop thread: pthread_join(), %s", strerror(result)); return -result; } } return 0; } int lo_server_thread_get_port(lo_server_thread st) { return lo_server_get_port(st->s); } char *lo_server_thread_get_url(lo_server_thread st) { return lo_server_get_url(st->s); } lo_server lo_server_thread_get_server(lo_server_thread st) { return st->s; } int lo_server_thread_events_pending(lo_server_thread st) { return lo_server_events_pending(st->s); } static void thread_func(void *data) { lo_server_thread st = (lo_server_thread)data; while (st->active) { lo_server_recv_noblock(st->s, 10); } st->done = 1; pthread_exit(NULL); } void lo_server_thread_pp(lo_server_thread st) { lo_server_pp(st->s); } /* vi:set ts=8 sts=4 sw=4: */