nunchuck: Initial implementation

This was tested with the Grove Nunchuck adapter.

Signed-off-by: Jon Trulson <jtrulson@ics.com>
Signed-off-by: Zion Orent <zorent@ics.com>
Signed-off-by: John Van Drasek <john.r.van.drasek@intel.com>
This commit is contained in:
Jon Trulson
2015-02-10 17:18:33 -07:00
committed by John Van Drasek
parent 780500ac0a
commit 1f3d074261
8 changed files with 461 additions and 0 deletions

View File

@ -0,0 +1,5 @@
set (libname "nunchuck")
set (libdescription "upm wii nunchuck module")
set (module_src ${libname}.cxx)
set (module_h ${libname}.h)
upm_module_init()

View File

@ -0,0 +1,8 @@
%module jsupm_nunchuck
%include "../upm.i"
%{
#include "nunchuck.h"
%}
%include "nunchuck.h"

143
src/nunchuck/nunchuck.cxx Normal file
View File

@ -0,0 +1,143 @@
/*
* Author: Jon Trulson <jtrulson@ics.com>
* Copyright (c) 2015 Intel Corporation.
*
* Permission is hereby granted, free of charge, to any person obtaining
* a copy of this software and associated documentation files (the
* "Software"), to deal in the Software without restriction, including
* without limitation the rights to use, copy, modify, merge, publish,
* distribute, sublicense, and/or sell copies of the Software, and to
* permit persons to whom the Software is furnished to do so, subject to
* the following conditions:
*
* The above copyright notice and this permission notice shall be
* included in all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
* LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
* OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
* WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
*/
#include <iostream>
#include <string>
#include <unistd.h>
#include "nunchuck.h"
using namespace upm;
using namespace std;
NUNCHUCK::NUNCHUCK(int bus, uint8_t addr)
{
stickX = 0;
stickY = 0;
accelX = 0;
accelY = 0;
accelZ = 0;
buttonC = false;
buttonZ = false;
// setup our i2c link
m_i2c = mraa_i2c_init(bus);
mraa_result_t rv;
if ( (rv = mraa_i2c_address(m_i2c, addr)) != MRAA_SUCCESS )
{
cerr << __FUNCTION__ << ": mraa_i2c_address() failed." << endl;
mraa_result_print(rv);
}
}
NUNCHUCK::~NUNCHUCK()
{
mraa_i2c_stop(m_i2c);
}
bool NUNCHUCK::writeByte(uint8_t reg, uint8_t byte)
{
mraa_result_t rv;
if ( (rv = mraa_i2c_write_byte_data(m_i2c, byte, reg)) != MRAA_SUCCESS )
{
cerr << __FUNCTION__ << ": mraa_i2c_write_byte_data() failed." << endl;
mraa_result_print(rv);
return false;
}
return true;
}
uint8_t NUNCHUCK::readBytes(uint8_t reg, uint8_t *buffer, unsigned int len)
{
if (!len || !buffer)
return 0;
mraa_i2c_address(m_i2c, NUNCHUCK_I2C_ADDR);
mraa_i2c_write_byte(m_i2c, reg);
return mraa_i2c_read(m_i2c, buffer, len);
}
bool NUNCHUCK::init()
{
usleep(1000000);
// disable encryption
if (!writeByte(0xf0, 0x55))
return false;
if (!writeByte(0xfb, 0x00))
return false;
return true;
}
void NUNCHUCK::update()
{
const int bufsize = 6;
uint8_t buf[bufsize];
int rv;
rv = readBytes(0x00, buf, bufsize);
if (rv != bufsize)
{
cerr << __FUNCTION__ << "read failed, expected " << bufsize
<< "bytes, got " << rv << endl;
return;
}
// analog stick X
stickX = buf[0];
// analog stick Y
stickY = buf[1];
// accelerometer X
accelX = ( (buf[2] << 2) | ((buf[5] & 0x0c) >> 2) );
// accelerometer Y
accelY = ( (buf[3] << 2) | ((buf[5] & 0x30) >> 4) );
// accelerometer Z
accelZ = ( (buf[4] << 2) | ((buf[5] & 0xc0) >> 6) );
// buttonC
if (buf[5] & 0x02)
buttonC = false;
else
buttonC = true;
// buttonZ
if (buf[5] & 0x01)
buttonZ = false;
else
buttonZ = true;
}

148
src/nunchuck/nunchuck.h Normal file
View File

@ -0,0 +1,148 @@
/*
* Author: Jon Trulson <jtrulson@ics.com>
* Copyright (c) 2015 Intel Corporation.
*
* Permission is hereby granted, free of charge, to any person obtaining
* a copy of this software and associated documentation files (the
* "Software"), to deal in the Software without restriction, including
* without limitation the rights to use, copy, modify, merge, publish,
* distribute, sublicense, and/or sell copies of the Software, and to
* permit persons to whom the Software is furnished to do so, subject to
* the following conditions:
*
* The above copyright notice and this permission notice shall be
* included in all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
* LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
* OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
* WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
*/
#pragma once
#include <string>
#include <mraa/i2c.h>
#define NUNCHUCK_I2C_ADDR 0x52
namespace upm {
/**
* @brief C++ API for the Wii Nunchuck
*
* UPM module for the Wii Nunchuck. This module was tested with the
* Wii Nunchuck connected to I2C via a Grove Wii Nunchuck adapter.
*
* See http://wiibrew.org/wiki/Wiimote/Extension_Controllers and
* http://wiibrew.org/wiki/Wiimote/Extension_Controllers/Nunchuck
* for more detailed information on the controller and it's protocol.
*
* A warning for the Grove nunchuck adapter: It has 2 traces on one
* side, and 3 traces on the other. Do not match these up with the
* nunchuck connector's traces. The nunchuck connector 'groove'
* should be on the same side as the grove interface socket on the
* adapter.
*
* @ingroup nunchuck
* @defgroup nunchuck libupm-nunchuck
* @snippet nunchuck.cxx Interesting
*/
class NUNCHUCK {
public:
/**
* NUNCHUCK constructor
*
* @param bus i2c bus to use
* @param addr i2c address to use
*/
NUNCHUCK(int bus, uint8_t addr=NUNCHUCK_I2C_ADDR);
/**
* NUNCHUCK destructor
*/
~NUNCHUCK();
/**
* Write value(s) into registers
*
* @param reg register location to start writing into
* @param byte byte to write
* @return true if successful
*/
bool writeByte(uint8_t reg, uint8_t byte);
/**
* Read value(s) from registers
*
* @param reg register location to start reading from
* @param buffer buffer for data storage
* @param len number of bytes to read
* @return number of bytes read
*/
uint8_t readBytes(uint8_t reg, uint8_t *buffer, unsigned int len);
/**
* Initialize the controller. Here, we disable encryption after
* delaying for a time to ensure the controller is ready.
*
* @return true if initialized successfully
*/
bool init();
/**
* Read and update the current state of the controller.
*
*/
void update();
/**
* Current analog stick X position
*
*/
int stickX;
/**
* Current analog stick Y position
*
*/
int stickY;
/**
* Current accelerometer X value
*
*/
int accelX;
/**
* Current accelerometer Y value
*
*/
int accelY;
/**
* Current accelerometer Z value
*
*/
int accelZ;
/**
* Button C pressed?
*
*/
bool buttonC;
/**
* Button Z pressed?
*
*/
bool buttonZ;
private:
mraa_i2c_context m_i2c;
};
}

View File

@ -0,0 +1,9 @@
%module pyupm_nunchuck
%include "../upm.i"
%feature("autodoc", "3");
%include "nunchuck.h"
%{
#include "nunchuck.h"
%}