mirror of
https://github.com/libretro/RetroArch
synced 2025-01-26 18:35:22 +00:00
70c85be081
per platform port that needs it
1107 lines
27 KiB
C
1107 lines
27 KiB
C
/*-------------------------------------------------------------
|
|
|
|
wpad.c -- Wiimote Application Programmers Interface
|
|
|
|
Copyright (C) 2008
|
|
Michael Wiedenbauer (shagkur)
|
|
Dave Murphy (WinterMute)
|
|
Hector Martin (marcan)
|
|
|
|
This software is provided 'as-is', without any express or implied
|
|
warranty. In no event will the authors be held liable for any
|
|
damages arising from the use of this software.
|
|
|
|
Permission is granted to anyone to use this software for any
|
|
purpose, including commercial applications, and to alter it and
|
|
redistribute it freely, subject to the following restrictions:
|
|
|
|
1. The origin of this software must not be misrepresented; you
|
|
must not claim that you wrote the original software. If you use
|
|
this software in a product, an acknowledgment in the product
|
|
documentation would be appreciated but is not required.
|
|
|
|
2. Altered source versions must be plainly marked as such, and
|
|
must not be misrepresented as being the original software.
|
|
|
|
3. This notice may not be removed or altered from any source
|
|
distribution.
|
|
|
|
-------------------------------------------------------------*/
|
|
|
|
/* This source as presented is a modified version of original wiiuse for use
|
|
* with RetroArch, and must not be confused with the original software. */
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <time.h>
|
|
#include <unistd.h>
|
|
|
|
#include "os.h"
|
|
#include "conf.h"
|
|
#include "ir.h"
|
|
|
|
#ifdef HAVE_WIIUSE_SPEAKER
|
|
#include "speaker.h"
|
|
#endif
|
|
|
|
#include "dynamics.h"
|
|
#include "wiiuse_internal.h"
|
|
#include "wiiuse/wpad.h"
|
|
#include "lwp_threads.h"
|
|
#include "ogcsys.h"
|
|
|
|
#define MAX_STREAMDATA_LEN 20
|
|
#define EVENTQUEUE_LENGTH 16
|
|
|
|
#define DISCONNECT_BATTERY_DIED 0x14
|
|
#define DISCONNECT_POWER_OFF 0x15
|
|
|
|
struct _wpad_thresh{
|
|
s32 btns;
|
|
s32 ir;
|
|
s32 js;
|
|
s32 acc;
|
|
s32 mp;
|
|
};
|
|
|
|
struct _wpad_cb {
|
|
wiimote *wm;
|
|
s32 data_fmt;
|
|
s32 queue_head;
|
|
s32 queue_tail;
|
|
s32 queue_full;
|
|
u32 queue_length;
|
|
u32 dropped_events;
|
|
s32 idle_time;
|
|
s32 speaker_enabled;
|
|
struct _wpad_thresh thresh;
|
|
|
|
#ifdef HAVE_WIIUSE_SPEAKER
|
|
void *sound_data;
|
|
u32 sound_len;
|
|
u32 sound_off;
|
|
syswd_t sound_alarm;
|
|
#endif
|
|
|
|
WPADData lstate;
|
|
#ifdef HAVE_WIIUSE_QUEUE_EXT
|
|
WPADData *queue_ext;
|
|
#endif
|
|
WPADData queue_int[EVENTQUEUE_LENGTH];
|
|
};
|
|
|
|
static syswd_t __wpad_timer;
|
|
static vu32 __wpads_inited = 0;
|
|
static vs32 __wpads_ponded = 0;
|
|
static u32 __wpad_idletimeout = 300;
|
|
static vu32 __wpads_active = 0;
|
|
static vu32 __wpads_used = 0;
|
|
static wiimote **__wpads = NULL;
|
|
static wiimote_listen __wpads_listen[CONF_PAD_MAX_REGISTERED];
|
|
static WPADData wpaddata[WPAD_MAX_WIIMOTES];
|
|
static struct _wpad_cb __wpdcb[WPAD_MAX_WIIMOTES];
|
|
static conf_pads __wpad_devs;
|
|
static struct linkkey_info __wpad_keys[WPAD_MAX_WIIMOTES];
|
|
|
|
static s32 __wpad_onreset(s32 final);
|
|
static s32 __wpad_disconnect(struct _wpad_cb *wpdcb);
|
|
static void __wpad_eventCB(struct wiimote_t *wm,s32 event);
|
|
|
|
static void __wpad_def_powcb(s32 chan);
|
|
static WPADShutdownCallback __wpad_batcb = NULL;
|
|
static WPADShutdownCallback __wpad_powcb = __wpad_def_powcb;
|
|
|
|
extern void __wiiuse_sensorbar_enable(int enable);
|
|
extern void __SYS_DoPowerCB(void);
|
|
|
|
static sys_resetinfo __wpad_resetinfo = {
|
|
{},
|
|
__wpad_onreset,
|
|
127
|
|
};
|
|
|
|
static s32 __wpad_onreset(s32 final)
|
|
{
|
|
//printf("__wpad_onreset(%d)\n",final);
|
|
if(final==FALSE) {
|
|
WPAD_Shutdown();
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
static void __wpad_def_powcb(s32 chan)
|
|
{
|
|
__SYS_DoPowerCB();
|
|
}
|
|
|
|
static void __wpad_timeouthandler(syswd_t alarm,void *cbarg)
|
|
{
|
|
s32 i;
|
|
struct wiimote_t *wm = NULL;
|
|
struct _wpad_cb *wpdcb = NULL;
|
|
|
|
if(!__wpads_active) return;
|
|
|
|
++_thread_dispatch_disable_level;
|
|
for(i=0;i<WPAD_MAX_WIIMOTES;i++) {
|
|
wpdcb = &__wpdcb[i];
|
|
wm = wpdcb->wm;
|
|
if(wm && WIIMOTE_IS_SET(wm,WIIMOTE_STATE_CONNECTED)) {
|
|
wpdcb->idle_time++;
|
|
if(wpdcb->idle_time>=__wpad_idletimeout) {
|
|
wpdcb->idle_time = 0;
|
|
wiiuse_disconnect(wm);
|
|
}
|
|
}
|
|
}
|
|
--_thread_dispatch_disable_level;
|
|
}
|
|
|
|
#ifdef HAVE_WIIUSE_SPEAKER
|
|
static void __wpad_sounddata_alarmhandler(syswd_t alarm,void *cbarg)
|
|
{
|
|
u8 *snd_data;
|
|
u32 snd_off;
|
|
struct wiimote_t *wm;
|
|
struct _wpad_cb *wpdcb = (struct _wpad_cb*)cbarg;
|
|
|
|
if(!wpdcb) return;
|
|
|
|
if(wpdcb->sound_off>=wpdcb->sound_len) {
|
|
wpdcb->sound_data = NULL;
|
|
wpdcb->sound_len = 0;
|
|
wpdcb->sound_off = 0;
|
|
SYS_CancelAlarm(wpdcb->sound_alarm);
|
|
return;
|
|
}
|
|
|
|
wm = wpdcb->wm;
|
|
snd_data = wpdcb->sound_data;
|
|
snd_off = wpdcb->sound_off;
|
|
wpdcb->sound_off += MAX_STREAMDATA_LEN;
|
|
wiiuse_write_streamdata(wm,(snd_data+snd_off),MAX_STREAMDATA_LEN,NULL);
|
|
}
|
|
#endif
|
|
|
|
static void __wpad_setfmt(s32 chan)
|
|
{
|
|
switch(__wpdcb[chan].data_fmt) {
|
|
case WPAD_FMT_BTNS:
|
|
wiiuse_set_flags(__wpads[chan], 0, WIIUSE_CONTINUOUS);
|
|
wiiuse_motion_sensing(__wpads[chan],0);
|
|
wiiuse_set_ir(__wpads[chan],0);
|
|
break;
|
|
case WPAD_FMT_BTNS_ACC:
|
|
wiiuse_set_flags(__wpads[chan], WIIUSE_CONTINUOUS, 0);
|
|
wiiuse_motion_sensing(__wpads[chan],1);
|
|
wiiuse_set_ir(__wpads[chan],0);
|
|
break;
|
|
case WPAD_FMT_BTNS_ACC_IR:
|
|
wiiuse_set_flags(__wpads[chan], WIIUSE_CONTINUOUS, 0);
|
|
wiiuse_motion_sensing(__wpads[chan],1);
|
|
wiiuse_set_ir(__wpads[chan],1);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
wiimote *__wpad_assign_slot(struct bd_addr *pad_addr)
|
|
{
|
|
u32 i, level;
|
|
struct bd_addr bdaddr;
|
|
//printf("WPAD Assigning slot (active: 0x%02x)\n", __wpads_used);
|
|
_CPU_ISR_Disable(level);
|
|
|
|
// Try preassigned slots
|
|
for(i=0; i<CONF_PAD_MAX_ACTIVE /*&& i<WPAD_MAX_WIIMOTES*/; i++) {
|
|
BD_ADDR(&(bdaddr),__wpad_devs.active[i].bdaddr[5],__wpad_devs.active[i].bdaddr[4],__wpad_devs.active[i].bdaddr[3],__wpad_devs.active[i].bdaddr[2],__wpad_devs.active[i].bdaddr[1],__wpad_devs.active[i].bdaddr[0]);
|
|
if(bd_addr_cmp(pad_addr,&bdaddr) && !(__wpads_used & (1<<i))) {
|
|
//printf("WPAD Got Preassigned Slot %d\n", i);
|
|
__wpads_used |= (0x01<<i);
|
|
_CPU_ISR_Restore(level);
|
|
return __wpads[i];
|
|
}
|
|
}
|
|
|
|
// No match, pick the first free slot
|
|
for(i=0; i<WPAD_MAX_WIIMOTES; i++) {
|
|
if(!(__wpads_used & (1<<i))) {
|
|
//printf("WPAD Got Free Slot %d\n", i);
|
|
__wpads_used |= (0x01<<i);
|
|
_CPU_ISR_Restore(level);
|
|
return __wpads[i];
|
|
}
|
|
}
|
|
//printf("WPAD All Slots Used\n");
|
|
_CPU_ISR_Restore(level);
|
|
return NULL;
|
|
}
|
|
|
|
static s32 __wpad_init_finished(s32 result,void *usrdata)
|
|
{
|
|
u32 i;
|
|
struct bd_addr bdaddr;
|
|
|
|
//printf("__wpad_init_finished(%d)\n",result);
|
|
|
|
if(result==ERR_OK) {
|
|
for(i=0;/*__wpads[i] && */i<__wpad_devs.num_registered;i++) {
|
|
BD_ADDR(&(bdaddr),__wpad_devs.registered[i].bdaddr[5],__wpad_devs.registered[i].bdaddr[4],__wpad_devs.registered[i].bdaddr[3],__wpad_devs.registered[i].bdaddr[2],__wpad_devs.registered[i].bdaddr[1],__wpad_devs.registered[i].bdaddr[0]);
|
|
wiiuse_register(&__wpads_listen[i],&(bdaddr),__wpad_assign_slot);
|
|
}
|
|
}
|
|
__wpads_inited = WPAD_STATE_ENABLED;
|
|
return ERR_OK;
|
|
}
|
|
|
|
static s32 __wpad_patch_finished(s32 result,void *usrdata)
|
|
{
|
|
//printf("__wpad_patch_finished(%d)\n",result);
|
|
BTE_InitSub(__wpad_init_finished);
|
|
return ERR_OK;
|
|
}
|
|
|
|
static s32 __readlinkkey_finished(s32 result,void *usrdata)
|
|
{
|
|
//printf("__readlinkkey_finished(%d)\n",result);
|
|
|
|
__wpads_ponded = result;
|
|
BTE_ApplyPatch(__wpad_patch_finished);
|
|
|
|
return ERR_OK;
|
|
}
|
|
|
|
static s32 __initcore_finished(s32 result,void *usrdata)
|
|
{
|
|
//printf("__initcore_finished(%d)\n",result);
|
|
|
|
if(result==ERR_OK) {
|
|
BTE_ReadStoredLinkKey(__wpad_keys,WPAD_MAX_WIIMOTES,__readlinkkey_finished);
|
|
}
|
|
return ERR_OK;
|
|
}
|
|
|
|
static s32 __wpad_disconnect(struct _wpad_cb *wpdcb)
|
|
{
|
|
struct wiimote_t *wm;
|
|
|
|
if(wpdcb==NULL) return 0;
|
|
|
|
wm = wpdcb->wm;
|
|
if(wm && WIIMOTE_IS_SET(wm,WIIMOTE_STATE_CONNECTED)) {
|
|
wiiuse_disconnect(wm);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void __wpad_calc_data(WPADData *data,WPADData *lstate,struct accel_t *accel_calib,u32 smoothed)
|
|
{
|
|
if(data->err!=WPAD_ERR_NONE) return;
|
|
|
|
data->orient = lstate->orient;
|
|
|
|
data->ir.state = lstate->ir.state;
|
|
data->ir.sensorbar = lstate->ir.sensorbar;
|
|
data->ir.x = lstate->ir.x;
|
|
data->ir.y = lstate->ir.y;
|
|
data->ir.sx = lstate->ir.sx;
|
|
data->ir.sy = lstate->ir.sy;
|
|
data->ir.ax = lstate->ir.ax;
|
|
data->ir.ay = lstate->ir.ay;
|
|
data->ir.distance = lstate->ir.distance;
|
|
data->ir.z = lstate->ir.z;
|
|
data->ir.angle = lstate->ir.angle;
|
|
data->ir.error_cnt = lstate->ir.error_cnt;
|
|
data->ir.glitch_cnt = lstate->ir.glitch_cnt;
|
|
|
|
if(data->data_present & WPAD_DATA_ACCEL) {
|
|
calculate_orientation(accel_calib, &data->accel, &data->orient, smoothed);
|
|
calculate_gforce(accel_calib, &data->accel, &data->gforce);
|
|
}
|
|
if(data->data_present & WPAD_DATA_IR) {
|
|
interpret_ir_data(&data->ir,&data->orient);
|
|
}
|
|
if(data->data_present & WPAD_DATA_EXPANSION) {
|
|
switch(data->exp.type) {
|
|
case EXP_NUNCHUK:
|
|
{
|
|
struct nunchuk_t *nc = &data->exp.nunchuk;
|
|
|
|
nc->orient = lstate->exp.nunchuk.orient;
|
|
calc_joystick_state(&nc->js,nc->js.pos.x,nc->js.pos.y);
|
|
calculate_orientation(&nc->accel_calib,&nc->accel,&nc->orient,smoothed);
|
|
calculate_gforce(&nc->accel_calib,&nc->accel,&nc->gforce);
|
|
data->btns_h |= (data->exp.nunchuk.btns<<16);
|
|
}
|
|
break;
|
|
|
|
case EXP_CLASSIC:
|
|
{
|
|
struct classic_ctrl_t *cc = &data->exp.classic;
|
|
|
|
cc->r_shoulder = ((f32)cc->rs_raw/0x1F);
|
|
cc->l_shoulder = ((f32)cc->ls_raw/0x1F);
|
|
calc_joystick_state(&cc->ljs, cc->ljs.pos.x, cc->ljs.pos.y);
|
|
calc_joystick_state(&cc->rjs, cc->rjs.pos.x, cc->rjs.pos.y);
|
|
data->btns_h |= (data->exp.classic.btns<<16);
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
*lstate = *data;
|
|
}
|
|
|
|
static void __save_state(struct wiimote_t* wm) {
|
|
/* wiimote */
|
|
wm->lstate.btns = wm->btns;
|
|
wm->lstate.accel = wm->accel;
|
|
|
|
/* ir */
|
|
wm->lstate.ir = wm->ir;
|
|
|
|
/* expansion */
|
|
switch (wm->exp.type) {
|
|
case EXP_NUNCHUK:
|
|
wm->lstate.exp.nunchuk = wm->exp.nunchuk;
|
|
break;
|
|
case EXP_CLASSIC:
|
|
wm->lstate.exp.classic = wm->exp.classic;
|
|
break;
|
|
case EXP_MOTION_PLUS:
|
|
wm->lstate.exp.mp = wm->exp.mp;
|
|
break;
|
|
}
|
|
}
|
|
|
|
#define ABS(x) ((s32)(x)>0?(s32)(x):-((s32)(x)))
|
|
|
|
#define STATE_CHECK(thresh, a, b) \
|
|
if(((thresh) > WPAD_THRESH_IGNORE) && (ABS((a)-(b)) > (thresh))) \
|
|
state_changed = 1;
|
|
|
|
#define STATE_CHECK_SIMPLE(thresh, a, b) \
|
|
if(((thresh) > WPAD_THRESH_IGNORE) && ((a) != (b))) \
|
|
state_changed = 1;
|
|
|
|
static u32 __wpad_read_expansion(struct wiimote_t *wm,WPADData *data, struct _wpad_thresh *thresh)
|
|
{
|
|
int state_changed = 0;
|
|
switch(data->exp.type) {
|
|
case EXP_NUNCHUK:
|
|
data->exp.nunchuk = wm->exp.nunchuk;
|
|
STATE_CHECK_SIMPLE(thresh->btns, wm->exp.nunchuk.btns, wm->lstate.exp.nunchuk.btns);
|
|
STATE_CHECK(thresh->acc, wm->exp.nunchuk.accel.x, wm->lstate.exp.nunchuk.accel.x);
|
|
STATE_CHECK(thresh->acc, wm->exp.nunchuk.accel.y, wm->lstate.exp.nunchuk.accel.y);
|
|
STATE_CHECK(thresh->acc, wm->exp.nunchuk.accel.z, wm->lstate.exp.nunchuk.accel.z);
|
|
STATE_CHECK(thresh->js, wm->exp.nunchuk.js.pos.x, wm->lstate.exp.nunchuk.js.pos.x);
|
|
STATE_CHECK(thresh->js, wm->exp.nunchuk.js.pos.y, wm->lstate.exp.nunchuk.js.pos.y);
|
|
break;
|
|
case EXP_CLASSIC:
|
|
data->exp.classic = wm->exp.classic;
|
|
STATE_CHECK_SIMPLE(thresh->btns, wm->exp.classic.btns, wm->lstate.exp.classic.btns);
|
|
STATE_CHECK(thresh->js, wm->exp.classic.rs_raw, wm->lstate.exp.classic.rs_raw);
|
|
STATE_CHECK(thresh->js, wm->exp.classic.ls_raw, wm->lstate.exp.classic.ls_raw);
|
|
STATE_CHECK(thresh->js, wm->exp.classic.ljs.pos.x, wm->lstate.exp.classic.ljs.pos.x);
|
|
STATE_CHECK(thresh->js, wm->exp.classic.ljs.pos.y, wm->lstate.exp.classic.ljs.pos.y);
|
|
STATE_CHECK(thresh->js, wm->exp.classic.rjs.pos.x, wm->lstate.exp.classic.rjs.pos.x);
|
|
STATE_CHECK(thresh->js, wm->exp.classic.rjs.pos.y, wm->lstate.exp.classic.rjs.pos.y);
|
|
break;
|
|
case EXP_MOTION_PLUS:
|
|
data->exp.mp = wm->exp.mp;
|
|
STATE_CHECK(thresh->mp,wm->exp.mp.rx,wm->lstate.exp.mp.rx);
|
|
STATE_CHECK(thresh->mp,wm->exp.mp.ry,wm->lstate.exp.mp.ry);
|
|
STATE_CHECK(thresh->mp,wm->exp.mp.rz,wm->lstate.exp.mp.rz);
|
|
break;
|
|
}
|
|
return state_changed;
|
|
}
|
|
|
|
static void __wpad_read_wiimote(struct wiimote_t *wm, WPADData *data, s32 *idle_time, struct _wpad_thresh *thresh)
|
|
{
|
|
int i;
|
|
int state_changed = 0;
|
|
data->err = WPAD_ERR_TRANSFER;
|
|
data->data_present = 0;
|
|
data->battery_level = wm->battery_level;
|
|
data->exp.type = wm->exp.type;
|
|
if(wm && WIIMOTE_IS_SET(wm,WIIMOTE_STATE_CONNECTED)) {
|
|
if(WIIMOTE_IS_SET(wm,WIIMOTE_STATE_HANDSHAKE_COMPLETE)) {
|
|
switch(wm->event_buf[0]) {
|
|
case WM_RPT_BTN:
|
|
case WM_RPT_BTN_ACC:
|
|
case WM_RPT_BTN_ACC_IR:
|
|
case WM_RPT_BTN_EXP:
|
|
case WM_RPT_BTN_ACC_EXP:
|
|
case WM_RPT_BTN_IR_EXP:
|
|
case WM_RPT_BTN_ACC_IR_EXP:
|
|
data->btns_h = (wm->btns&0xffff);
|
|
data->data_present |= WPAD_DATA_BUTTONS;
|
|
STATE_CHECK_SIMPLE(thresh->btns, wm->btns, wm->lstate.btns);
|
|
}
|
|
switch(wm->event_buf[0]) {
|
|
case WM_RPT_BTN_ACC:
|
|
case WM_RPT_BTN_ACC_IR:
|
|
case WM_RPT_BTN_ACC_EXP:
|
|
case WM_RPT_BTN_ACC_IR_EXP:
|
|
data->accel = wm->accel;
|
|
data->data_present |= WPAD_DATA_ACCEL;
|
|
STATE_CHECK(thresh->acc, wm->accel.x, wm->lstate.accel.x);
|
|
STATE_CHECK(thresh->acc, wm->accel.y, wm->lstate.accel.y);
|
|
STATE_CHECK(thresh->acc, wm->accel.z, wm->lstate.accel.z);
|
|
}
|
|
switch(wm->event_buf[0]) {
|
|
//IR requires acceleration
|
|
//case WM_RPT_BTN_IR_EXP:
|
|
case WM_RPT_BTN_ACC_IR:
|
|
case WM_RPT_BTN_ACC_IR_EXP:
|
|
data->ir = wm->ir;
|
|
data->data_present |= WPAD_DATA_IR;
|
|
for(i=0; i<WPAD_MAX_IR_DOTS; i++) {
|
|
STATE_CHECK_SIMPLE(thresh->ir, wm->ir.dot[i].visible, wm->lstate.ir.dot[i].visible);
|
|
STATE_CHECK(thresh->ir, wm->ir.dot[i].rx, wm->lstate.ir.dot[i].rx);
|
|
STATE_CHECK(thresh->ir, wm->ir.dot[i].ry, wm->lstate.ir.dot[i].ry);
|
|
}
|
|
}
|
|
switch(wm->event_buf[0]) {
|
|
case WM_RPT_BTN_EXP:
|
|
case WM_RPT_BTN_ACC_EXP:
|
|
case WM_RPT_BTN_IR_EXP:
|
|
case WM_RPT_BTN_ACC_IR_EXP:
|
|
state_changed |= __wpad_read_expansion(wm,data,thresh);
|
|
data->data_present |= WPAD_DATA_EXPANSION;
|
|
}
|
|
data->err = WPAD_ERR_NONE;
|
|
if(state_changed) {
|
|
*idle_time = 0;
|
|
__save_state(wm);
|
|
}
|
|
} else
|
|
data->err = WPAD_ERR_NOT_READY;
|
|
} else
|
|
data->err = WPAD_ERR_NO_CONTROLLER;
|
|
}
|
|
|
|
static void __wpad_eventCB(struct wiimote_t *wm,s32 event)
|
|
{
|
|
s32 chan;
|
|
u32 maxbufs;
|
|
WPADData *wpadd = NULL;
|
|
struct _wpad_cb *wpdcb = NULL;
|
|
|
|
switch(event) {
|
|
case WIIUSE_EVENT:
|
|
chan = wm->unid;
|
|
wpdcb = &__wpdcb[chan];
|
|
|
|
#ifdef HAVE_WIIUSE_QUEUE_EXT
|
|
if(wpdcb->queue_ext!=NULL) {
|
|
maxbufs = wpdcb->queue_length;
|
|
wpadd = &(wpdcb->queue_ext[wpdcb->queue_tail]);
|
|
}
|
|
else
|
|
#endif
|
|
{
|
|
maxbufs = EVENTQUEUE_LENGTH;
|
|
wpadd = &(wpdcb->queue_int[wpdcb->queue_tail]);
|
|
}
|
|
if(wpdcb->queue_full == maxbufs) {
|
|
wpdcb->queue_head++;
|
|
wpdcb->queue_head %= maxbufs;
|
|
wpdcb->dropped_events++;
|
|
} else {
|
|
wpdcb->queue_full++;
|
|
}
|
|
|
|
__wpad_read_wiimote(wm, wpadd, &wpdcb->idle_time, &wpdcb->thresh);
|
|
|
|
wpdcb->queue_tail++;
|
|
wpdcb->queue_tail %= maxbufs;
|
|
|
|
break;
|
|
case WIIUSE_STATUS:
|
|
break;
|
|
case WIIUSE_CONNECT:
|
|
chan = wm->unid;
|
|
wpdcb = &__wpdcb[chan];
|
|
wpdcb->wm = wm;
|
|
wpdcb->queue_head = 0;
|
|
wpdcb->queue_tail = 0;
|
|
wpdcb->queue_full = 0;
|
|
wpdcb->idle_time = 0;
|
|
memset(&wpdcb->lstate,0,sizeof(WPADData));
|
|
memset(&wpaddata[chan],0,sizeof(WPADData));
|
|
memset(wpdcb->queue_int,0,(sizeof(WPADData)*EVENTQUEUE_LENGTH));
|
|
wiiuse_set_ir_position(wm,(CONF_GetSensorBarPosition()^1));
|
|
wiiuse_set_ir_sensitivity(wm,CONF_GetIRSensitivity());
|
|
wiiuse_set_leds(wm,(WIIMOTE_LED_1<<(chan%WPAD_BALANCE_BOARD)),NULL);
|
|
#ifdef HAVE_WIIUSE_SPEAKER
|
|
wiiuse_set_speaker(wm,wpdcb->speaker_enabled);
|
|
#endif
|
|
__wpad_setfmt(chan);
|
|
__wpads_active |= (0x01<<chan);
|
|
break;
|
|
case WIIUSE_DISCONNECT:
|
|
chan = wm->unid;
|
|
wpdcb = &__wpdcb[chan];
|
|
wpdcb->wm = wm;
|
|
wpdcb->queue_head = 0;
|
|
wpdcb->queue_tail = 0;
|
|
wpdcb->queue_full = 0;
|
|
wpdcb->queue_length = 0;
|
|
#ifdef HAVE_WIIUSE_QUEUE_EXT
|
|
wpdcb->queue_ext = NULL;
|
|
#endif
|
|
wpdcb->idle_time = -1;
|
|
memset(&wpdcb->lstate,0,sizeof(WPADData));
|
|
memset(&wpaddata[chan],0,sizeof(WPADData));
|
|
memset(wpdcb->queue_int,0,(sizeof(WPADData)*EVENTQUEUE_LENGTH));
|
|
__wpads_active &= ~(0x01<<chan);
|
|
__wpads_used &= ~(0x01<<chan);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
void __wpad_disconnectCB(struct bd_addr *offaddr, u8 reason)
|
|
{
|
|
struct bd_addr bdaddr;
|
|
int i;
|
|
|
|
if(__wpads_inited == WPAD_STATE_ENABLED) {
|
|
for(i=0;i<__wpad_devs.num_registered;i++) {
|
|
BD_ADDR(&(bdaddr),__wpad_devs.registered[i].bdaddr[5],__wpad_devs.registered[i].bdaddr[4],__wpad_devs.registered[i].bdaddr[3],__wpad_devs.registered[i].bdaddr[2],__wpad_devs.registered[i].bdaddr[1],__wpad_devs.registered[i].bdaddr[0]);
|
|
if(bd_addr_cmp(offaddr,&bdaddr)) {
|
|
if(reason == DISCONNECT_BATTERY_DIED) {
|
|
if(__wpad_batcb) __wpad_batcb(i); //sanity check since this pointer can be NULL.
|
|
} else if(reason == DISCONNECT_POWER_OFF)
|
|
__wpad_powcb(i); //no sanity check because there's a default callback iff not otherwise set.
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
s32 WPAD_Init()
|
|
{
|
|
u32 level;
|
|
struct timespec tb;
|
|
int i;
|
|
|
|
_CPU_ISR_Disable(level);
|
|
if(__wpads_inited==WPAD_STATE_DISABLED) {
|
|
__wpads_ponded = 0;
|
|
__wpads_active = 0;
|
|
|
|
memset(__wpdcb,0,sizeof(struct _wpad_cb)*WPAD_MAX_WIIMOTES);
|
|
memset(&__wpad_devs,0,sizeof(conf_pads));
|
|
memset(__wpad_keys,0,sizeof(struct linkkey_info)*WPAD_MAX_WIIMOTES);
|
|
|
|
for(i=0;i<WPAD_MAX_WIIMOTES;i++) {
|
|
__wpdcb[i].thresh.btns = WPAD_THRESH_DEFAULT_BUTTONS;
|
|
__wpdcb[i].thresh.ir = WPAD_THRESH_DEFAULT_IR;
|
|
__wpdcb[i].thresh.acc = WPAD_THRESH_DEFAULT_ACCEL;
|
|
__wpdcb[i].thresh.js = WPAD_THRESH_DEFAULT_JOYSTICK;
|
|
__wpdcb[i].thresh.mp = WPAD_THRESH_DEFAULT_MOTION_PLUS;
|
|
|
|
#ifdef HAVE_WIIUSE_SPEAKER
|
|
if (SYS_CreateAlarm(&__wpdcb[i].sound_alarm) < 0)
|
|
{
|
|
WPAD_Shutdown();
|
|
_CPU_ISR_Restore(level);
|
|
return WPAD_ERR_UNKNOWN;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
if(CONF_GetPadDevices(&__wpad_devs) < 0) {
|
|
WPAD_Shutdown();
|
|
_CPU_ISR_Restore(level);
|
|
return WPAD_ERR_BADCONF;
|
|
}
|
|
|
|
if(__wpad_devs.num_registered == 0) {
|
|
WPAD_Shutdown();
|
|
_CPU_ISR_Restore(level);
|
|
return WPAD_ERR_NONEREGISTERED;
|
|
}
|
|
|
|
if(__wpad_devs.num_registered > CONF_PAD_MAX_REGISTERED) {
|
|
WPAD_Shutdown();
|
|
_CPU_ISR_Restore(level);
|
|
return WPAD_ERR_BADCONF;
|
|
}
|
|
|
|
__wpads = wiiuse_init(WPAD_MAX_WIIMOTES,__wpad_eventCB);
|
|
if(__wpads==NULL) {
|
|
WPAD_Shutdown();
|
|
_CPU_ISR_Restore(level);
|
|
return WPAD_ERR_UNKNOWN;
|
|
}
|
|
|
|
__wiiuse_sensorbar_enable(1);
|
|
|
|
BTE_Init();
|
|
BTE_SetDisconnectCallback(__wpad_disconnectCB);
|
|
BTE_InitCore(__initcore_finished);
|
|
|
|
if (SYS_CreateAlarm(&__wpad_timer) < 0)
|
|
{
|
|
WPAD_Shutdown();
|
|
_CPU_ISR_Restore(level);
|
|
return WPAD_ERR_UNKNOWN;
|
|
}
|
|
|
|
SYS_RegisterResetFunc(&__wpad_resetinfo);
|
|
|
|
tb.tv_sec = 1;
|
|
tb.tv_nsec = 0;
|
|
SYS_SetPeriodicAlarm(__wpad_timer,&tb,&tb,__wpad_timeouthandler,NULL);
|
|
__wpads_inited = WPAD_STATE_ENABLING;
|
|
}
|
|
_CPU_ISR_Restore(level);
|
|
return WPAD_ERR_NONE;
|
|
}
|
|
|
|
s32 WPAD_ReadEvent(s32 chan, WPADData *data)
|
|
{
|
|
u32 level;
|
|
u32 maxbufs,smoothed = 0;
|
|
struct accel_t *accel_calib = NULL;
|
|
struct _wpad_cb *wpdcb = NULL;
|
|
WPADData *lstate = NULL,*wpadd = NULL;
|
|
|
|
//if(chan<WPAD_CHAN_0 || chan>=WPAD_MAX_WIIMOTES) return WPAD_ERR_BAD_CHANNEL;
|
|
|
|
_CPU_ISR_Disable(level);
|
|
if(__wpads_inited==WPAD_STATE_DISABLED) {
|
|
_CPU_ISR_Restore(level);
|
|
return WPAD_ERR_NOT_READY;
|
|
}
|
|
|
|
if(__wpads[chan] && WIIMOTE_IS_SET(__wpads[chan],WIIMOTE_STATE_CONNECTED)) {
|
|
if(WIIMOTE_IS_SET(__wpads[chan],WIIMOTE_STATE_HANDSHAKE_COMPLETE)) {
|
|
wpdcb = &__wpdcb[chan];
|
|
#ifdef HAVE_WIIUSE_QUEUE_EXT
|
|
if(wpdcb->queue_ext!=NULL) {
|
|
maxbufs = wpdcb->queue_length;
|
|
wpadd = wpdcb->queue_ext;
|
|
}
|
|
else
|
|
#endif
|
|
{
|
|
maxbufs = EVENTQUEUE_LENGTH;
|
|
wpadd = wpdcb->queue_int;
|
|
}
|
|
if(wpdcb->queue_full == 0) {
|
|
_CPU_ISR_Restore(level);
|
|
return WPAD_ERR_QUEUE_EMPTY;
|
|
}
|
|
if(data)
|
|
*data = wpadd[wpdcb->queue_head];
|
|
wpdcb->queue_head++;
|
|
wpdcb->queue_head %= maxbufs;
|
|
wpdcb->queue_full--;
|
|
lstate = &wpdcb->lstate;
|
|
accel_calib = &__wpads[chan]->accel_calib;
|
|
smoothed = WIIMOTE_IS_FLAG_SET(__wpads[chan], WIIUSE_SMOOTHING);
|
|
} else {
|
|
_CPU_ISR_Restore(level);
|
|
return WPAD_ERR_NOT_READY;
|
|
}
|
|
} else {
|
|
_CPU_ISR_Restore(level);
|
|
return WPAD_ERR_NO_CONTROLLER;
|
|
}
|
|
|
|
_CPU_ISR_Restore(level);
|
|
if(data)
|
|
__wpad_calc_data(data,lstate,accel_calib,smoothed);
|
|
return 0;
|
|
}
|
|
|
|
static s32 pad_readpending_temp(s32 chan)
|
|
{
|
|
s32 count = 0;
|
|
s32 ret;
|
|
|
|
while(1)
|
|
{
|
|
ret = WPAD_ReadEvent(chan, &wpaddata[chan]);
|
|
if(ret < WPAD_ERR_NONE)
|
|
break;
|
|
count++;
|
|
}
|
|
if(ret == WPAD_ERR_QUEUE_EMPTY) return count;
|
|
return ret;
|
|
}
|
|
|
|
s32 WPAD_ReadPending(s32 chan, WPADDataCallback datacb)
|
|
{
|
|
u32 i;
|
|
s32 count = 0;
|
|
s32 ret;
|
|
|
|
for(i= WPAD_CHAN_0; i < WPAD_MAX_WIIMOTES; i++)
|
|
if((ret = pad_readpending_temp(i)) >= WPAD_ERR_NONE)
|
|
count += ret;
|
|
return count;
|
|
}
|
|
|
|
s32 WPAD_SetMotionPlus(s32 chan, u8 enable)
|
|
{
|
|
u32 level;
|
|
s32 ret;
|
|
int i;
|
|
|
|
if(chan == WPAD_CHAN_ALL) {
|
|
for(i=WPAD_CHAN_0; i<WPAD_MAX_WIIMOTES; i++)
|
|
if((ret = WPAD_SetMotionPlus(i, enable)) < WPAD_ERR_NONE)
|
|
return ret;
|
|
return WPAD_ERR_NONE;
|
|
}
|
|
|
|
if(chan<WPAD_CHAN_0 || chan>=WPAD_MAX_WIIMOTES) return WPAD_ERR_BAD_CHANNEL;
|
|
|
|
_CPU_ISR_Disable(level);
|
|
if(__wpads_inited==WPAD_STATE_DISABLED) {
|
|
_CPU_ISR_Restore(level);
|
|
return WPAD_ERR_NOT_READY;
|
|
}
|
|
|
|
if(__wpads[chan]!=NULL) {
|
|
wiiuse_set_motion_plus(__wpads[chan], enable);
|
|
}
|
|
_CPU_ISR_Restore(level);
|
|
return WPAD_ERR_NONE;
|
|
}
|
|
|
|
s32 WPAD_SetVRes(s32 chan,u32 xres,u32 yres)
|
|
{
|
|
u32 level;
|
|
s32 ret;
|
|
int i;
|
|
|
|
if(chan == WPAD_CHAN_ALL) {
|
|
for(i=WPAD_CHAN_0; i<WPAD_MAX_WIIMOTES; i++)
|
|
if((ret = WPAD_SetVRes(i, xres, yres)) < WPAD_ERR_NONE)
|
|
return ret;
|
|
return WPAD_ERR_NONE;
|
|
}
|
|
|
|
if(chan<WPAD_CHAN_0 || chan>=WPAD_MAX_WIIMOTES) return WPAD_ERR_BAD_CHANNEL;
|
|
|
|
_CPU_ISR_Disable(level);
|
|
if(__wpads_inited==WPAD_STATE_DISABLED) {
|
|
_CPU_ISR_Restore(level);
|
|
return WPAD_ERR_NOT_READY;
|
|
}
|
|
|
|
if(__wpads[chan]!=NULL)
|
|
wiiuse_set_ir_vres(__wpads[chan],xres,yres);
|
|
|
|
_CPU_ISR_Restore(level);
|
|
return WPAD_ERR_NONE;
|
|
}
|
|
|
|
s32 WPAD_GetStatus()
|
|
{
|
|
s32 ret;
|
|
u32 level;
|
|
|
|
_CPU_ISR_Disable(level);
|
|
ret = __wpads_inited;
|
|
_CPU_ISR_Restore(level);
|
|
|
|
return ret;
|
|
}
|
|
|
|
s32 WPAD_Probe(s32 chan,u32 *type)
|
|
{
|
|
s32 ret;
|
|
u32 level,dev;
|
|
wiimote *wm = NULL;
|
|
|
|
//if(chan<WPAD_CHAN_0 || chan>=WPAD_MAX_WIIMOTES) return WPAD_ERR_BAD_CHANNEL;
|
|
|
|
_CPU_ISR_Disable(level);
|
|
if(__wpads_inited==WPAD_STATE_DISABLED) {
|
|
_CPU_ISR_Restore(level);
|
|
return WPAD_ERR_NOT_READY;
|
|
}
|
|
|
|
wm = __wpads[chan];
|
|
if(wm && WIIMOTE_IS_SET(wm,WIIMOTE_STATE_CONNECTED)) {
|
|
if(WIIMOTE_IS_SET(wm,WIIMOTE_STATE_HANDSHAKE_COMPLETE)) {
|
|
dev = WPAD_EXP_NONE;
|
|
if(WIIMOTE_IS_SET(wm,WIIMOTE_STATE_EXP)) {
|
|
switch(wm->exp.type) {
|
|
case WPAD_EXP_NUNCHUK:
|
|
case WPAD_EXP_CLASSIC:
|
|
dev = wm->exp.type;
|
|
break;
|
|
}
|
|
}
|
|
if(type!=NULL) *type = dev;
|
|
ret = WPAD_ERR_NONE;
|
|
} else
|
|
ret = WPAD_ERR_NOT_READY;
|
|
} else
|
|
ret = WPAD_ERR_NO_CONTROLLER;
|
|
_CPU_ISR_Restore(level);
|
|
|
|
return ret;
|
|
}
|
|
|
|
#ifdef HAVE_WIIUSE_QUEUE_EXT
|
|
s32 WPAD_SetEventBufs(s32 chan, WPADData *bufs, u32 cnt)
|
|
{
|
|
u32 level;
|
|
struct _wpad_cb *wpdcb = NULL;
|
|
|
|
if(chan<WPAD_CHAN_0 || chan>=WPAD_MAX_WIIMOTES) return WPAD_ERR_BAD_CHANNEL;
|
|
|
|
_CPU_ISR_Disable(level);
|
|
wpdcb = &__wpdcb[chan];
|
|
wpdcb->queue_head = 0;
|
|
wpdcb->queue_tail = 0;
|
|
wpdcb->queue_full = 0;
|
|
wpdcb->queue_length = cnt;
|
|
wpdcb->queue_ext = bufs;
|
|
_CPU_ISR_Restore(level);
|
|
return WPAD_ERR_NONE;
|
|
}
|
|
#endif
|
|
|
|
void WPAD_SetPowerButtonCallback(WPADShutdownCallback cb)
|
|
{
|
|
u32 level;
|
|
|
|
_CPU_ISR_Disable(level);
|
|
if(cb)
|
|
__wpad_powcb = cb;
|
|
else
|
|
__wpad_powcb = __wpad_def_powcb;
|
|
_CPU_ISR_Restore(level);
|
|
}
|
|
|
|
void WPAD_SetBatteryDeadCallback(WPADShutdownCallback cb)
|
|
{
|
|
u32 level;
|
|
|
|
_CPU_ISR_Disable(level);
|
|
__wpad_batcb = cb;
|
|
_CPU_ISR_Restore(level);
|
|
}
|
|
|
|
s32 WPAD_Disconnect(s32 chan)
|
|
{
|
|
u32 level, cnt = 0;
|
|
struct _wpad_cb *wpdcb = NULL;
|
|
|
|
if(chan<WPAD_CHAN_0 || chan>=WPAD_MAX_WIIMOTES) return WPAD_ERR_BAD_CHANNEL;
|
|
|
|
_CPU_ISR_Disable(level);
|
|
if(__wpads_inited==WPAD_STATE_DISABLED) {
|
|
_CPU_ISR_Restore(level);
|
|
return WPAD_ERR_NOT_READY;
|
|
}
|
|
|
|
wpdcb = &__wpdcb[chan];
|
|
__wpad_disconnect(wpdcb);
|
|
|
|
_CPU_ISR_Restore(level);
|
|
|
|
while(__wpads_active&(0x01<<chan)) {
|
|
usleep(50);
|
|
if(++cnt > 3000) break;
|
|
}
|
|
|
|
return WPAD_ERR_NONE;
|
|
}
|
|
|
|
void WPAD_Shutdown()
|
|
{
|
|
s32 i;
|
|
u32 level;
|
|
u32 cnt = 0;
|
|
struct _wpad_cb *wpdcb = NULL;
|
|
|
|
_CPU_ISR_Disable(level);
|
|
|
|
__wpads_inited = WPAD_STATE_DISABLED;
|
|
SYS_RemoveAlarm(__wpad_timer);
|
|
for(i=0;i<WPAD_MAX_WIIMOTES;i++) {
|
|
wpdcb = &__wpdcb[i];
|
|
#ifdef HAVE_WIIUSE_SPEAKER
|
|
SYS_RemoveAlarm(wpdcb->sound_alarm);
|
|
#endif
|
|
__wpad_disconnect(wpdcb);
|
|
}
|
|
|
|
__wiiuse_sensorbar_enable(0);
|
|
_CPU_ISR_Restore(level);
|
|
|
|
while(__wpads_active) {
|
|
usleep(50);
|
|
if(++cnt > 3000) break;
|
|
}
|
|
|
|
BTE_Shutdown();
|
|
}
|
|
|
|
void WPAD_SetIdleTimeout(u32 seconds)
|
|
{
|
|
u32 level;
|
|
|
|
_CPU_ISR_Disable(level);
|
|
__wpad_idletimeout = seconds;
|
|
_CPU_ISR_Restore(level);
|
|
}
|
|
|
|
#ifdef HAVE_WIIUSE_RUMBLE
|
|
s32 WPAD_Rumble(s32 chan, int status)
|
|
{
|
|
int i;
|
|
s32 ret;
|
|
u32 level;
|
|
|
|
if(chan == WPAD_CHAN_ALL) {
|
|
for(i=WPAD_CHAN_0; i<WPAD_MAX_WIIMOTES; i++)
|
|
if((ret = WPAD_Rumble(i,status)) < WPAD_ERR_NONE)
|
|
return ret;
|
|
return WPAD_ERR_NONE;
|
|
}
|
|
|
|
if(chan<WPAD_CHAN_0 || chan>=WPAD_MAX_WIIMOTES) return WPAD_ERR_BAD_CHANNEL;
|
|
|
|
_CPU_ISR_Disable(level);
|
|
if(__wpads_inited==WPAD_STATE_DISABLED) {
|
|
_CPU_ISR_Restore(level);
|
|
return WPAD_ERR_NOT_READY;
|
|
}
|
|
|
|
if(__wpads[chan]!=NULL)
|
|
wiiuse_rumble(__wpads[chan],status);
|
|
|
|
_CPU_ISR_Restore(level);
|
|
return WPAD_ERR_NONE;
|
|
}
|
|
#endif
|
|
|
|
#ifdef HAVE_WIIUSE_SPEAKER
|
|
s32 WPAD_ControlSpeaker(s32 chan,s32 enable)
|
|
{
|
|
int i;
|
|
s32 ret;
|
|
u32 level;
|
|
|
|
if(chan == WPAD_CHAN_ALL) {
|
|
for(i=WPAD_CHAN_0; i<WPAD_MAX_WIIMOTES; i++)
|
|
if((ret = WPAD_ControlSpeaker(i,enable)) < WPAD_ERR_NONE)
|
|
return ret;
|
|
return WPAD_ERR_NONE;
|
|
}
|
|
|
|
if(chan<WPAD_CHAN_0 || chan>=WPAD_MAX_WIIMOTES) return WPAD_ERR_BAD_CHANNEL;
|
|
|
|
_CPU_ISR_Disable(level);
|
|
if(__wpads_inited==WPAD_STATE_DISABLED) {
|
|
_CPU_ISR_Restore(level);
|
|
return WPAD_ERR_NOT_READY;
|
|
}
|
|
|
|
if(__wpads[chan]!=NULL) {
|
|
__wpdcb[chan].speaker_enabled = enable;
|
|
wiiuse_set_speaker(__wpads[chan],enable);
|
|
}
|
|
|
|
_CPU_ISR_Restore(level);
|
|
return WPAD_ERR_NONE;
|
|
}
|
|
|
|
s32 WPAD_IsSpeakerEnabled(s32 chan)
|
|
{
|
|
s32 ret;
|
|
u32 level;
|
|
wiimote *wm = NULL;
|
|
|
|
if(chan<WPAD_CHAN_0 || chan>=WPAD_MAX_WIIMOTES) return WPAD_ERR_BAD_CHANNEL;
|
|
|
|
_CPU_ISR_Disable(level);
|
|
if(__wpads_inited==WPAD_STATE_DISABLED) {
|
|
_CPU_ISR_Restore(level);
|
|
return WPAD_ERR_NOT_READY;
|
|
}
|
|
|
|
wm = __wpads[chan];
|
|
ret = WPAD_ERR_NOT_READY;
|
|
if(wm && WIIMOTE_IS_SET(wm,WIIMOTE_STATE_CONNECTED)) {
|
|
if(WIIMOTE_IS_SET(wm,WIIMOTE_STATE_HANDSHAKE_COMPLETE)
|
|
&& WIIMOTE_IS_SET(wm,WIIMOTE_STATE_SPEAKER)) ret = WPAD_ERR_NONE;
|
|
}
|
|
|
|
_CPU_ISR_Restore(level);
|
|
return ret;
|
|
}
|
|
|
|
void WPAD_EncodeData(WPADEncStatus *info,u32 flag,const s16 *pcmSamples,s32 numSamples,u8 *encData)
|
|
{
|
|
int n;
|
|
short *samples = (short*)pcmSamples;
|
|
WENCStatus *status = (WENCStatus*)info;
|
|
|
|
if(!(flag&WPAD_ENC_CONT)) status->step = 0;
|
|
|
|
n = (numSamples+1)/2;
|
|
for(;n>0;n--) {
|
|
int nibble;
|
|
nibble = (wencdata(status,samples[0]))<<4;
|
|
nibble |= (wencdata(status,samples[1]));
|
|
*encData++ = nibble;
|
|
samples += 2;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
WPADData *WPAD_Data(int chan)
|
|
{
|
|
//if(chan<0 || chan>=WPAD_MAX_WIIMOTES) return NULL;
|
|
return &wpaddata[chan];
|
|
}
|
|
|
|
u8 WPAD_BatteryLevel(int chan)
|
|
{
|
|
//if(chan<0 || chan>=WPAD_MAX_WIIMOTES) return 0;
|
|
return wpaddata[chan].battery_level;
|
|
}
|
|
|
|
void WPAD_IR(int chan, struct ir_t *ir)
|
|
{
|
|
//if(chan<0 || chan>=WPAD_MAX_WIIMOTES || ir==NULL ) return;
|
|
*ir = wpaddata[chan].ir;
|
|
}
|
|
|
|
void WPAD_Orientation(int chan, struct orient_t *orient)
|
|
{
|
|
//if(chan<0 || chan>=WPAD_MAX_WIIMOTES || orient==NULL ) return;
|
|
*orient = wpaddata[chan].orient;
|
|
}
|
|
|
|
void WPAD_GForce(int chan, struct gforce_t *gforce)
|
|
{
|
|
//if(chan<0 || chan>=WPAD_MAX_WIIMOTES || gforce==NULL ) return;
|
|
*gforce = wpaddata[chan].gforce;
|
|
}
|
|
|
|
void WPAD_Accel(int chan, struct vec3w_t *accel)
|
|
{
|
|
//if(chan<0 || chan>=WPAD_MAX_WIIMOTES || accel==NULL ) return;
|
|
*accel = wpaddata[chan].accel;
|
|
}
|