Hi all, I am wondering if anyone can shed some light on the following predicament. I am by no means a multi-threading guru so any insight would be most appreciated.
The following are relevant excerpts from the code of an external. AFAIK the external initializes mutex and cond and spawns a secondary worker thread that deals with audio-unfriendly (xrun-causing) write operations to the wiimote and terminates it when the object is destructed waiting for the thread to join back and then destroying the mutex. Now, if I add a bit of usleep right after the thread has been spawned as part of the constructor (as included below) the external seems very stable (e.g. cutting and pasting it as fast as keyboard allows, or in other words constructing and destructing instances of it as fast as possible does not result in a crash). Yet, when one does not use usleep right after spawning the secondary (worker) thread in the constructor, the whole thing is very crash-prone, almost as if the spawning of thread does not go well unless given adequate time to do get things all into sync, so to say, even though this makes to me no sense as the way I understand it the constructor does not move ahead until pthread_create does not return a value (which in this case I am not bothering to read). Curiously, when not using usleep, a crash may occur right at creation time, at any point while the object exists, and even as late as during its destruction. Any ideas? P.S. I am also including the entire file for those interested in trying it out. Best wishes, Ico Relevant excerpts (in random order and incomplete to allow for greater legibility): //struct defining the object typedef struct _wiimote { t_object x_obj; // standard pd object (must be first in struct) ... //Creating separate threads for actions known to cause sample drop-outs pthread_t unsafe_t; pthread_mutex_t unsafe_mutex; pthread_cond_t unsafe_cond; t_float unsafe; ... t_float led; ... } t_wiimote; //constructor static void *pd_cwiid_new(t_symbol* s, int argc, t_atom *argv) { ... x->led = 0; // spawn threads for actions known to cause sample drop-outs threadedFunctionParams rPars; rPars.wiimote = x; pthread_mutex_init(&x->unsafe_mutex, NULL); pthread_cond_init(&x->unsafe_cond, NULL); pthread_create( &x->unsafe_t, NULL, (void *) &pd_cwiid_pthreadForAudioUnfriendlyOperations, (void *) &rPars); //WHY IS THIS NECESSARY? I thought that pthread_create call will first finish spawning thread before proceeding usleep(100); //allow thread to sync (is there a better way to do this?) ... } //destructor static void pd_cwiid_free(t_wiimote* x) { if (x->connected) { pd_cwiid_doDisconnect(x); //this one has nothing to do with thread but rather disconnects the wiimote } x->unsafe = -1; //to allow secondary thread to exit the while loop pthread_mutex_lock(&x->unsafe_mutex); pthread_cond_signal(&x->unsafe_cond); pthread_mutex_unlock(&x->unsafe_mutex); pthread_join(x->unsafe_t, NULL); pthread_mutex_destroy(&x->unsafe_mutex); ... } //worker thread void pd_cwiid_pthreadForAudioUnfriendlyOperations(void *ptr) { threadedFunctionParams *rPars = (threadedFunctionParams*)ptr; t_wiimote *x = rPars->wiimote; t_float local_led = 0; t_float local_rumble = 0; unsigned char local_rpt_mode = x->rpt_mode; while(x->unsafe > -1) { pthread_mutex_lock(&x->unsafe_mutex); if ((local_led == x->led) && (local_rumble == x->rumble) && (local_rpt_mode == x->rpt_mode)) { pthread_cond_wait(&x->unsafe_cond, &x->unsafe_mutex); } if (local_led != x->led) { local_led = x->led; //do something } } if (local_rumble != x->rumble) { local_rumble = x->rumble; //do something else } ... pthread_mutex_unlock(&x->unsafe_mutex); } pthread_exit(0); } //an example of how the thread is affected by the main thread void pd_cwiid_setLED(t_wiimote *x, t_floatarg f) { if (x->connected) { x->led = f; pthread_mutex_lock(&x->unsafe_mutex); pthread_cond_signal(&x->unsafe_cond); pthread_mutex_unlock(&x->unsafe_mutex); } }
// =================================================================== // Wiimote external for Puredata // Written by Mike Wozniewki (Feb 2007), www.mikewoz.com // // Requires the CWiid library (version 0.6.00) by L. Donnie Smith // // =================================================================== // This program 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 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 General Public License for more details. // // You should have received a copy of the GNU 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 // =================================================================== // ChangeLog: // 2008-04-14 Florian Krebs // * adapt wiimote external for the actual version of cwiid (0.6.00) // ChangeLog: // 2009-06-09 DISIS (Michael Hawthorne <rustm...@gmail.com> & Ivica Ico Bukvic <i...@vt.edu>) // http://disis.music.vt.edu // * Bug-fixes (connecting and disconnecting crashes) // * Multithreaded implementation to prevent wiimote from starving PD audio thread // * Bang implementation to allow for better data rate control // * Updated help file // v 0.6.3 Changelog: // 2009-10-05 DISIS (Ivica Ico Bukvic <i...@vt.edu>) // http://disis.music.vt.edu // * Total rewrite of the threaded design and tons of clean-up // v0.6.4 Changelog: // 2010-03-21 DISIS (Ivica Ico Bukvic <i...@vt.edu>) // http://disis.music.vt.edu // * Reworked signalling system to use clock_delay() // v0.6.5 Changelog: // 2010-09-15 DISIS (Ivica Ico Bukvic <i...@vt.edu>) // http://disis.music.vt.edu // * Added motionplus support // * Squashed bugs where expansion was not recognized on connect // * Fixed incorrect calibration on connect // * Other minor bugs'n'fixes #include <stdio.h> #include <unistd.h> #include <sys/select.h> #include <bluetooth/bluetooth.h> #include <m_pd.h> #include <math.h> #include <pthread.h> #include "cwiid_internal.h" #define PI 3.14159265358979323 //#define DARWIN_CALIB struct acc { unsigned char x; unsigned char y; unsigned char z; }; /* Wiimote Callback */ cwiid_mesg_callback_t pd_cwiid_callback; // class and struct declarations for wiimote pd external: static t_class *pd_cwiid_class; typedef struct _wiimote { t_object x_obj; // standard pd object (must be first in struct) cwiid_wiimote_t *wiimote; // individual wiimote handle per pd object, represented in libcwiid t_float connected; int wiimoteID; int extensionAttached; //Creating separate threads for actions known to cause sample drop-outs pthread_t unsafe_t; pthread_mutex_t unsafe_mutex; pthread_cond_t unsafe_cond; t_float unsafe; t_float rumble; t_float led; t_float rpt; unsigned char rpt_mode; t_symbol *addr; t_float toggle_acc, toggle_ir, toggle_nc; struct acc acc_zero, acc_one; // acceleration struct acc nc_acc_zero, nc_acc_one; // nunchuck acceleration // We store atom list for each data type so we don't waste time // allocating memory at every callback: t_atom btn_atoms[2]; t_atom old_btn_atoms[2]; t_atom acc_atoms[3]; t_atom ir_atoms[4]; t_atom nc_btn_atoms[1]; t_atom old_nc_btn_atoms[1]; t_atom nc_acc_atoms[3]; t_atom nc_stick_atoms[2]; t_atom mp_acc_atoms[3]; //motionplus //for thread-unsafe operations t_clock *x_clock; // outlets: t_outlet *outlet_btn; t_outlet *outlet_acc; t_outlet *outlet_ir; t_outlet *outlet_nc_btn; t_outlet *outlet_nc_acc; t_outlet *outlet_nc_stick; t_outlet *outlet_mp_acc; t_outlet *outlet_connected; } t_wiimote; // For now, we make one global t_wiimote pointer that we can refer to // in the cwiid_callback. This means we can support maximum of ONE // wiimote. ARGH. We'll have to figure out how to have access to the // pd object from the callback (without modifying the CWiid code): #define MAX_WIIMOTES 1 t_wiimote *g_wiimoteList[MAX_WIIMOTES]; // Structure to pass generic parameters into a threaded function. // Added by VT DISIS typedef struct { t_wiimote *wiimote; } threadedFunctionParams; // ============================================================== void pd_cwiid_tick(t_wiimote *x) { outlet_float(x->outlet_connected, x->connected); } void pd_cwiid_debug(t_wiimote *x) { post("\n======================"); if (x->connected) post("Wiimote (id: %d) is connected.", x->wiimoteID); else post("Wiimote (id: %d) is NOT connected.", x->wiimoteID); if (x->toggle_acc) post("acceleration: ON"); else post("acceleration: OFF"); if (x->toggle_ir) post("IR: ON"); else post("IR: OFF"); if (x->toggle_nc) post("Nunchuck: ON"); else post("Nunchuck: OFF"); post(""); post("Accelerometer calibration: zero=(%d,%d,%d) one=(%d,%d,%d)",x->acc_zero.x,x->acc_zero.y,x->acc_zero.z,x->acc_one.x,x->acc_one.y,x->acc_one.z); post("Nunchuck calibration: zero=(%d,%d,%d) one=(%d,%d,%d)",x->nc_acc_zero.x,x->nc_acc_zero.y,x->nc_acc_zero.z,x->nc_acc_one.x,x->nc_acc_one.y,x->nc_acc_one.z); } // ============================================================== // Button handler: void pd_cwiid_btn(t_wiimote *x, struct cwiid_btn_mesg *mesg) { //post("Buttons: %X %X", (mesg->buttons & 0xFF00)>>8, mesg->buttons & 0x00FF); SETFLOAT(x->btn_atoms+0, (mesg->buttons & 0xFF00)>>8); SETFLOAT(x->btn_atoms+1, mesg->buttons & 0x00FF); //outlet_anything(x->outlet_btn, &s_list, 2, x->btn_atoms); /* if (mesg->buttons & CWIID_BTN_UP) {} if (mesg->buttons & CWIID_BTN_DOWN) {} if (mesg->buttons & CWIID_BTN_LEFT) {} if (mesg->buttons & CWIID_BTN_RIGHT) {} if (mesg->buttons & CWIID_BTN_A) {} if (mesg->buttons & CWIID_BTN_B) {} if (mesg->buttons & CWIID_BTN_MINUS) {} if (mesg->buttons & CWIID_BTN_PLUS) {} if (mesg->buttons & CWIID_BTN_HOME) {} if (mesg->buttons & CWIID_BTN_1) {} if (mesg->buttons & CWIID_BTN_2) {} */ } // Records acceleration into wiimote object. // To retrieve the information in pd, send a bang to input or change output mode to 1 void pd_cwiid_acc(t_wiimote *x, struct cwiid_acc_mesg *mesg) { if (x->toggle_acc) { double a_x, a_y, a_z; a_x = ((double)mesg->acc[CWIID_X] - x->acc_zero.x) / (x->acc_one.x - x->acc_zero.x); a_y = ((double)mesg->acc[CWIID_Y] - x->acc_zero.y) / (x->acc_one.y - x->acc_zero.y); a_z = ((double)mesg->acc[CWIID_Z] - x->acc_zero.z) / (x->acc_one.z - x->acc_zero.z); SETFLOAT(x->acc_atoms+0, a_x); SETFLOAT(x->acc_atoms+1, a_y); SETFLOAT(x->acc_atoms+2, a_z); } } void pd_cwiid_ir(t_wiimote *x, struct cwiid_ir_mesg *mesg) { unsigned int i; if (x->toggle_ir) { //post("IR (valid,x,y,size) #%d: %d %d %d %d", i, data->ir_data.ir_src[i].valid, data->ir_data.ir_src[i].x, data->ir_data.ir_src[i].y, data->ir_data.ir_src[i].size); for (i=0; i<CWIID_IR_SRC_COUNT; i++) { if (mesg->src[i].valid) { SETFLOAT(x->ir_atoms+0, i); SETFLOAT(x->ir_atoms+1, mesg->src[i].pos[CWIID_X]); SETFLOAT(x->ir_atoms+2, mesg->src[i].pos[CWIID_Y]); SETFLOAT(x->ir_atoms+3, mesg->src[i].size); } } } } void pd_cwiid_nunchuk(t_wiimote *x, struct cwiid_nunchuk_mesg *mesg) { double a_x, a_y, a_z; a_x = ((double)mesg->acc[CWIID_X] - x->nc_acc_zero.x) / (x->nc_acc_one.x - x->nc_acc_zero.x); a_y = ((double)mesg->acc[CWIID_Y] - x->nc_acc_zero.y) / (x->nc_acc_one.y - x->nc_acc_zero.y); a_z = ((double)mesg->acc[CWIID_Z] - x->nc_acc_zero.z) / (x->nc_acc_one.z - x->nc_acc_zero.z); //if (mesg->buttons & CWIID_NUNCHUK_BTN_C) {} if (atom_getint(x->nc_btn_atoms) != mesg->buttons) { SETFLOAT(x->nc_btn_atoms+0, mesg->buttons); } SETFLOAT(x->nc_acc_atoms+0, a_x); SETFLOAT(x->nc_acc_atoms+1, a_y); SETFLOAT(x->nc_acc_atoms+2, a_z); SETFLOAT(x->nc_stick_atoms+0, mesg->stick[CWIID_X]); SETFLOAT(x->nc_stick_atoms+1, mesg->stick[CWIID_Y]); } void pd_cwiid_motionplus(t_wiimote *x, struct cwiid_motionplus_mesg *mesg) { //static gchar str[LBLVAL_LEN]; SETFLOAT(x->mp_acc_atoms+0, (double)mesg->angle_rate[CWIID_PHI]); SETFLOAT(x->mp_acc_atoms+1, (double)mesg->angle_rate[CWIID_THETA]); SETFLOAT(x->mp_acc_atoms+2, (double)mesg->angle_rate[CWIID_PSI]); // if (gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(chkExt))) { // gtk_widget_modify_bg(evMPPhiSlow, GTK_STATE_NORMAL, // (mesg->low_speed[CWIID_PHI]) ? &btn_on : &btn_off); // gtk_widget_modify_bg(evMPThetaSlow, GTK_STATE_NORMAL, // (mesg->low_speed[CWIID_THETA]) ? &btn_on : &btn_off); // gtk_widget_modify_bg(evMPPsiSlow, GTK_STATE_NORMAL, // (mesg->low_speed[CWIID_PSI]) ? &btn_on : &btn_off); // // //g_snprintf(str, LBLVAL_LEN, "%X", mesg->angle_rate[CWIID_PHI]); // gtk_label_set_text(GTK_LABEL(lblMPPhiVal), str); // gtk_progress_bar_set_fraction(GTK_PROGRESS_BAR(progMPPhi), // (double)mesg->angle_rate[CWIID_PHI]/0x4000); // g_snprintf(str, LBLVAL_LEN, "%X", mesg->angle_rate[CWIID_THETA]); // gtk_label_set_text(GTK_LABEL(lblMPThetaVal), str); // gtk_progress_bar_set_fraction(GTK_PROGRESS_BAR(progMPTheta), // (double)mesg->angle_rate[CWIID_THETA]/0x4000); // g_snprintf(str, LBLVAL_LEN, "%X", mesg->angle_rate[CWIID_PSI]); // gtk_label_set_text(GTK_LABEL(lblMPPsiVal), str); // gtk_progress_bar_set_fraction(GTK_PROGRESS_BAR(progMPPsi), // (double)mesg->angle_rate[CWIID_PSI]/0x4000); // } } void pd_cwiid_doBang(t_wiimote *x) { if (x->toggle_nc == 1 && x->extensionAttached == 1) { outlet_anything(x->outlet_nc_stick, &s_list, 2, x->nc_stick_atoms); outlet_anything(x->outlet_nc_acc, &s_list, 3, x->nc_acc_atoms); //motionplus outlet_anything(x->outlet_mp_acc, &s_list, 3, x->mp_acc_atoms); if (atom_getfloat(x->old_nc_btn_atoms) != atom_getfloat(x->nc_btn_atoms)) { outlet_float(x->outlet_nc_btn, atom_getfloat(x->nc_btn_atoms)); SETFLOAT(x->old_nc_btn_atoms+0, atom_getfloat(x->nc_btn_atoms)); } } if (x->toggle_ir == 1) outlet_anything(x->outlet_ir, &s_list, 4, x->ir_atoms); if (x->toggle_acc == 1) outlet_anything(x->outlet_acc, &s_list, 3, x->acc_atoms); if (x->connected) { if (atom_getfloat(x->old_btn_atoms+0) != atom_getfloat(x->btn_atoms+0) || atom_getfloat(x->old_btn_atoms+1) != atom_getfloat(x->btn_atoms+1)) { outlet_anything(x->outlet_btn, &s_list, 2, x->btn_atoms); SETFLOAT(x->old_btn_atoms+0, atom_getfloat(x->btn_atoms+0)); SETFLOAT(x->old_btn_atoms+1, atom_getfloat(x->btn_atoms+1)); } } } void pd_cwiid_status(t_wiimote *x) { if (x->connected ) { cwiid_request_status(x->wiimote); } } void pd_cwiid_setRumble(t_wiimote *x, t_floatarg f) { if (x->connected) { x->rumble = f; pthread_mutex_lock(&x->unsafe_mutex); pthread_cond_signal(&x->unsafe_cond); pthread_mutex_unlock(&x->unsafe_mutex); } } // The CWiid library invokes a callback function whenever events are // generated by the wiimote. This function is specified when connecting // to the wiimote (in the cwiid_open function). // Unfortunately, the mesg struct passed as an argument to the // callback does not have a pointer to the wiimote instance, and it // is thus impossible to know which wiimote has invoked the callback. // For this case we provide a hard-coded set of wrapper callbacks to // indicate which Pd wiimote instance to control. // So far I have only checked with one wiimote void pd_cwiid_callback(cwiid_wiimote_t *wiimote, int mesg_count, union cwiid_mesg mesg_array[], struct timespec *timestamp) { int i; t_wiimote *x; static unsigned char buf[CWIID_MAX_READ_LEN]; if (g_wiimoteList[wiimote->id] == NULL) { post("no wiimote loaded: %d%",wiimote->id); } else { x = g_wiimoteList[wiimote->id]; for (i=0; i < mesg_count; i++) { switch (mesg_array[i].type) { case CWIID_MESG_STATUS: post("Battery: %d%", (int) (100.0 * mesg_array[i].status_mesg.battery / CWIID_BATTERY_MAX)); switch (mesg_array[i].status_mesg.ext_type) { case CWIID_EXT_NONE: post("No nunchuck attached"); x->extensionAttached = 0; break; case CWIID_EXT_NUNCHUK: post("Nunchuck extension attached"); x->extensionAttached = 1; #ifdef DARWIN_CALIB x->nc_acc_zero.x = 128; x->nc_acc_zero.y = 129; x->nc_acc_zero.z = 128; x->nc_acc_one.x = 153; x->nc_acc_one.y = 154; x->nc_acc_one.z = 154; #else if (cwiid_read(x->wiimote, CWIID_RW_REG | CWIID_RW_DECODE, 0xA40020, 7, buf)) { post("Unable to retrieve Nunchuk calibration"); } else { x->nc_acc_zero.x = buf[0]; x->nc_acc_zero.y = buf[1]; x->nc_acc_zero.z = buf[2]; x->nc_acc_one.x = buf[4]; x->nc_acc_one.y = buf[5]; x->nc_acc_one.z = buf[6]; } #endif break; case CWIID_EXT_CLASSIC: post("Classic controller attached. There is no support for this yet."); break; case CWIID_EXT_MOTIONPLUS: x->extensionAttached = 1; post("MotionPlus extension attached"); break; case CWIID_EXT_BALANCE: post("Balance board attached. There is no support for this yet."); break; case CWIID_EXT_UNKNOWN: post("Unknown extension attached"); break; } break; case CWIID_MESG_BTN: pd_cwiid_btn(x, &mesg_array[i].btn_mesg); break; case CWIID_MESG_ACC: pd_cwiid_acc(x, &mesg_array[i].acc_mesg); break; case CWIID_MESG_IR: pd_cwiid_ir(x, &mesg_array[i].ir_mesg); break; case CWIID_MESG_NUNCHUK: pd_cwiid_nunchuk(x, &mesg_array[i].nunchuk_mesg); break; case CWIID_MESG_CLASSIC: //pd_cwiid_classic(x, &mesg_array[i].classic_mesg); //todo break; case CWIID_MESG_MOTIONPLUS: pd_cwiid_motionplus(x, &mesg_array[i].motionplus_mesg); break; default: break; } } } } // ============================================================== void pd_cwiid_setReportMode(t_wiimote *x, t_floatarg r) { if (x->connected) { if (r >= 0) x->rpt_mode = (unsigned char) r; else { x->rpt_mode = CWIID_RPT_STATUS | CWIID_RPT_BTN; if (x->toggle_ir) x->rpt_mode |= CWIID_RPT_IR; if (x->toggle_acc) x->rpt_mode |= CWIID_RPT_ACC; if (x->toggle_nc) x->rpt_mode |= CWIID_RPT_EXT; } pthread_mutex_lock(&x->unsafe_mutex); pthread_cond_signal(&x->unsafe_cond); pthread_mutex_unlock(&x->unsafe_mutex); } } void pd_cwiid_reportAcceleration(t_wiimote *x, t_floatarg f) { x->toggle_acc = f; pd_cwiid_setReportMode(x, -1); } void pd_cwiid_reportIR(t_wiimote *x, t_floatarg f) { x->toggle_ir = f; pd_cwiid_setReportMode(x, -1); } void pd_cwiid_reportExpansion(t_wiimote *x, t_floatarg f) { x->toggle_nc = f; pd_cwiid_setReportMode(x, -1); //if (x->connected) { // cwiid_enable(x->wiimote, CWIID_FLAG_MOTIONPLUS); // cwiid_request_status(x->wiimote); //} } void pd_cwiid_pthreadForAudioUnfriendlyOperations(void *ptr) { threadedFunctionParams *rPars = (threadedFunctionParams*)ptr; t_wiimote *x = rPars->wiimote; t_float local_led = 0; t_float local_rumble = 0; unsigned char local_rpt_mode = x->rpt_mode; while(x->unsafe > -1) { pthread_mutex_lock(&x->unsafe_mutex); if ((local_led == x->led) && (local_rumble == x->rumble) && (local_rpt_mode == x->rpt_mode)) { pthread_cond_wait(&x->unsafe_cond, &x->unsafe_mutex); } if (local_led != x->led) { local_led = x->led; // some possible values: // CWIID_LED0_ON 0x01 // CWIID_LED1_ON 0x02 // CWIID_LED2_ON 0x04 // CWIID_LED3_ON 0x08 if (cwiid_command(x->wiimote, CWIID_CMD_LED, local_led)) { //post("wiiremote error: problem setting LED."); } } if (local_rumble != x->rumble) { local_rumble = x->rumble; if (cwiid_command(x->wiimote, CWIID_CMD_RUMBLE, local_rumble)) { //post("wiiremote error: problem setting rumble."); } } if (local_rpt_mode != x->rpt_mode) { local_rpt_mode = x->rpt_mode; if (cwiid_command(x->wiimote, CWIID_CMD_RPT_MODE, local_rpt_mode)) { //post("wiimote error: problem setting report mode."); } } pthread_mutex_unlock(&x->unsafe_mutex); } pthread_exit(0); } void pd_cwiid_setLED(t_wiimote *x, t_floatarg f) { if (x->connected) { x->led = f; pthread_mutex_lock(&x->unsafe_mutex); pthread_cond_signal(&x->unsafe_cond); pthread_mutex_unlock(&x->unsafe_mutex); } } // The following function attempts to connect to a wiimote at a // specific address, provided as an argument. eg, 00:19:1D:70:CE:72 // This address can be discovered by running the following command // in a console: // hcitool scan | grep Nintendo void pd_cwiid_doConnect(t_wiimote *x, t_symbol *addr) { if (!x->connected) { int i; bdaddr_t bdaddr; static unsigned char buf[CWIID_MAX_READ_LEN]; x->addr = addr; // determine address: if (x->addr==gensym("NULL")) { post("Searching automatically..."); bdaddr = *BDADDR_ANY; } else { str2ba(x->addr->s_name, &bdaddr); post("Connecting to given address..."); post("Press buttons 1 and 2 simultaneously."); } // connect: for (i=0;i<MAX_WIIMOTES;++i) { if (g_wiimoteList[i]==NULL) { post("open: Connect wiimote %d",i); x->wiimote = cwiid_open(&bdaddr,CWIID_FLAG_MESG_IFC); x->wiimoteID = i; if (x->wiimote) { x->wiimote->id = i; g_wiimoteList[i] = x; } break; } } if (x->wiimote == NULL) { post("Error: could not find and/or connect to a wiimote. Please ensure that bluetooth is enabled, and that the 'hcitool scan' command lists your Nintendo device."); } else { post("wiimote has successfully connected"); #ifdef DARWIN_CALIB x->acc_zero.x = 128; x->acc_zero.y = 129; x->acc_zero.z = 128; x->acc_one.x = 153; x->acc_one.y = 154; x->acc_one.z = 154; #else if (cwiid_read(x->wiimote, CWIID_RW_EEPROM, 0x16, 7, buf)) { post("Unable to retrieve accelerometer calibration"); } else { x->acc_zero.x = buf[0]; x->acc_zero.y = buf[1]; x->acc_zero.z = buf[2]; x->acc_one.x = buf[4]; x->acc_one.y = buf[5]; x->acc_one.z = buf[6]; //post("Retrieved wiimote calibration: zero=(%.1f,%.1f,%.1f) one=(%.1f,%.1f,%.1f)",buf[0],buf[2],buf[3],buf[4],buf[5],buf[6]); } #endif if (cwiid_set_mesg_callback(x->wiimote, &pd_cwiid_callback)) { post("Connection error: Unable to set message callback"); } else { x->connected = 1; pd_cwiid_setReportMode(x,-1); cwiid_enable(x->wiimote, CWIID_FLAG_MOTIONPLUS); cwiid_request_status(x->wiimote); clock_delay(x->x_clock, 0); SETFLOAT(x->btn_atoms+0, 0); SETFLOAT(x->btn_atoms+1, 0); SETFLOAT(x->nc_btn_atoms+0, 0); SETFLOAT(x->old_btn_atoms+0, 0); SETFLOAT(x->old_btn_atoms+1, 0); SETFLOAT(x->old_nc_btn_atoms+0, 0); // send brief rumble to acknowledge connect // and give a bit of a wait before doing so usleep(500000); pd_cwiid_setRumble(x, 1); usleep(250000); pd_cwiid_setRumble(x, 0); } } } } // The following function attempts to discover a wiimote. It requires // that the user puts the wiimote into 'discoverable' mode before being // called. This is done by pressing the red button under the battery // cover, or by pressing buttons 1 and 2 simultaneously. void pd_cwiid_discover(t_wiimote *x) { if (!x->connected) { post("Put the wiimote into discover mode by pressing buttons 1 and 2 simultaneously."); pd_cwiid_doConnect(x, gensym("NULL")); } else { post("connect: device already connected!"); } } void pd_cwiid_doDisconnect(t_wiimote *x) { if (x->connected) { //cwiid_disable(x->wiimote, CWIID_FLAG_MOTIONPLUS); if (cwiid_close(x->wiimote)) { post("wiimote error: problems when disconnecting."); } else { post("disconnect successful, resetting values"); // reinitialize values: g_wiimoteList[x->wiimoteID] = NULL; x->toggle_acc = 0; x->toggle_ir = 0; x->toggle_nc = 0; SETFLOAT(x->acc_atoms+0, 0); SETFLOAT(x->acc_atoms+1, 0); SETFLOAT(x->acc_atoms+2, 0); SETFLOAT(x->ir_atoms+0, 0); SETFLOAT(x->ir_atoms+1, 0); SETFLOAT(x->ir_atoms+2, 0); SETFLOAT(x->ir_atoms+3, 0); SETFLOAT(x->nc_acc_atoms+0, 0); SETFLOAT(x->nc_acc_atoms+1, 0); SETFLOAT(x->nc_acc_atoms+2, 0); SETFLOAT(x->nc_stick_atoms+0, 0); SETFLOAT(x->nc_stick_atoms+1, 0); SETFLOAT(x->mp_acc_atoms+0, 0); SETFLOAT(x->mp_acc_atoms+1, 0); SETFLOAT(x->mp_acc_atoms+2, 0); x->connected = 0; x->wiimoteID = -1; x->extensionAttached = 0; x->addr = gensym("NULL"); x->wiimote = NULL; // x->rpt = 0; // x->unsafe = 0; // x->rumble = 0; // x->led = 0; // x->rpt_mode = -1; //signal disconnect on the outlet clock_delay(x->x_clock, 0); } } else post("disconnect: device is not connected!"); } // ============================================================== // ============================================================== static void *pd_cwiid_new(t_symbol* s, int argc, t_atom *argv) { post( "DISIS threaded implementation of wiimote object v.0.6.5"); //bdaddr_t bdaddr; // wiimote bdaddr t_wiimote *x = (t_wiimote *)pd_new(pd_cwiid_class); // create outlets: x->outlet_btn = outlet_new(&x->x_obj, &s_list); x->outlet_acc = outlet_new(&x->x_obj, &s_list); x->outlet_ir = outlet_new(&x->x_obj, &s_list); x->outlet_nc_btn = outlet_new(&x->x_obj, &s_float); x->outlet_nc_acc = outlet_new(&x->x_obj, &s_list); x->outlet_nc_stick = outlet_new(&x->x_obj, &s_list); x->outlet_mp_acc = outlet_new(&x->x_obj, &s_list); // status outlet: x->outlet_connected = outlet_new(&x->x_obj, &s_float); // initialize toggles: x->toggle_acc = 0; x->toggle_ir = 0; x->toggle_nc = 0; // initialize values: SETFLOAT(x->acc_atoms+0, 0); SETFLOAT(x->acc_atoms+1, 0); SETFLOAT(x->acc_atoms+2, 0); SETFLOAT(x->ir_atoms+0, 0); SETFLOAT(x->ir_atoms+1, 0); SETFLOAT(x->ir_atoms+2, 0); SETFLOAT(x->ir_atoms+3, 0); SETFLOAT(x->nc_acc_atoms+0, 0); SETFLOAT(x->nc_acc_atoms+1, 0); SETFLOAT(x->nc_acc_atoms+2, 0); SETFLOAT(x->nc_stick_atoms+0, 0); SETFLOAT(x->nc_stick_atoms+1, 0); SETFLOAT(x->mp_acc_atoms+0, 0); SETFLOAT(x->mp_acc_atoms+1, 0); SETFLOAT(x->mp_acc_atoms+2, 0); x->connected = 0; x->wiimoteID = -1; x->extensionAttached = 0; x->rumble = 0; x->led = 0; x->addr = gensym("NULL"); x->rpt = 0; x->unsafe = 0; x->rpt_mode = -1; x->x_clock = clock_new(x, (t_method)pd_cwiid_tick); // spawn threads for actions known to cause sample drop-outs threadedFunctionParams rPars; rPars.wiimote = x; pthread_mutex_init(&x->unsafe_mutex, NULL); pthread_cond_init(&x->unsafe_cond, NULL); pthread_create( &x->unsafe_t, NULL, (void *) &pd_cwiid_pthreadForAudioUnfriendlyOperations, (void *) &rPars); usleep(100); //allow thread to sync (is there a better way to do this?) // connect if user provided an address as an argument: if (argc==2) { post("conecting to provided address..."); if (argv->a_type == A_SYMBOL) { pd_cwiid_doConnect(x, atom_getsymbol(argv)); } else { error("[wiimote] expects either no argument, or a bluetooth address as an argument. eg, 00:19:1D:70:CE:72"); return NULL; } } return (x); } static void pd_cwiid_free(t_wiimote* x) { if (x->connected) { pd_cwiid_doDisconnect(x); } x->unsafe = -1; pthread_mutex_lock(&x->unsafe_mutex); pthread_cond_signal(&x->unsafe_cond); pthread_mutex_unlock(&x->unsafe_mutex); pthread_join(x->unsafe_t, NULL); pthread_mutex_destroy(&x->unsafe_mutex); clock_free(x->x_clock); } void disis_wiimote_setup(void) { int i; for (i=0; i<MAX_WIIMOTES; i++) g_wiimoteList[i] = NULL; pd_cwiid_class = class_new(gensym("disis_wiimote"), (t_newmethod)pd_cwiid_new, (t_method)pd_cwiid_free, sizeof(t_wiimote), CLASS_DEFAULT, A_GIMME, 0); class_addmethod(pd_cwiid_class, (t_method) pd_cwiid_debug, gensym("debug"), 0); class_addmethod(pd_cwiid_class, (t_method) pd_cwiid_doConnect, gensym("connect"), A_SYMBOL, 0); class_addmethod(pd_cwiid_class, (t_method) pd_cwiid_doDisconnect, gensym("disconnect"), 0); class_addmethod(pd_cwiid_class, (t_method) pd_cwiid_discover, gensym("discover"), 0); class_addmethod(pd_cwiid_class, (t_method) pd_cwiid_setReportMode, gensym("setReportMode"), A_DEFFLOAT, 0); class_addmethod(pd_cwiid_class, (t_method) pd_cwiid_reportAcceleration, gensym("reportAcceleration"), A_DEFFLOAT, 0); class_addmethod(pd_cwiid_class, (t_method) pd_cwiid_reportExpansion, gensym("reportExpansion"), A_DEFFLOAT, 0); class_addmethod(pd_cwiid_class, (t_method) pd_cwiid_reportIR, gensym("reportIR"), A_DEFFLOAT, 0); class_addmethod(pd_cwiid_class, (t_method) pd_cwiid_setRumble, gensym("setRumble"), A_DEFFLOAT, 0); class_addmethod(pd_cwiid_class, (t_method) pd_cwiid_setLED, gensym("setLED"), A_DEFFLOAT, 0); class_addmethod(pd_cwiid_class, (t_method) pd_cwiid_status, gensym("status"), 0); class_addbang(pd_cwiid_class, pd_cwiid_doBang); }
_______________________________________________ Linux-audio-dev mailing list Linux-audio-dev@lists.linuxaudio.org http://lists.linuxaudio.org/listinfo/linux-audio-dev