165 lines
3.8 KiB
C
165 lines
3.8 KiB
C
|
/*
|
||
|
* mcp23016.c:
|
||
|
* Extend wiringPi with the MCP 23016 I2C GPIO expander chip
|
||
|
* Copyright (c) 2013 Gordon Henderson
|
||
|
***********************************************************************
|
||
|
* This file is part of wiringPi:
|
||
|
* https://projects.drogon.net/raspberry-pi/wiringpi/
|
||
|
*
|
||
|
* wiringPi 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 3 of the
|
||
|
* License, or (at your option) any later version.
|
||
|
*
|
||
|
* wiringPi 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.
|
||
|
*
|
||
|
* You should have received a copy of the GNU Lesser General Public
|
||
|
* License along with wiringPi.
|
||
|
* If not, see <http://www.gnu.org/licenses/>.
|
||
|
***********************************************************************
|
||
|
*/
|
||
|
|
||
|
#include <stdio.h>
|
||
|
#include <pthread.h>
|
||
|
|
||
|
#include "wiringPi.h"
|
||
|
#include "wiringPiI2C.h"
|
||
|
#include "mcp23016.h"
|
||
|
|
||
|
#include "mcp23016reg.h"
|
||
|
|
||
|
|
||
|
/*
|
||
|
* myPinMode:
|
||
|
*********************************************************************************
|
||
|
*/
|
||
|
|
||
|
static void myPinMode (struct wiringPiNodeStruct *node, int pin, int mode)
|
||
|
{
|
||
|
int mask, old, reg ;
|
||
|
|
||
|
pin -= node->pinBase ;
|
||
|
|
||
|
if (pin < 8) // Bank A
|
||
|
reg = MCP23016_IODIR0 ;
|
||
|
else
|
||
|
{
|
||
|
reg = MCP23016_IODIR1 ;
|
||
|
pin &= 0x07 ;
|
||
|
}
|
||
|
|
||
|
mask = 1 << pin ;
|
||
|
old = wiringPiI2CReadReg8 (node->fd, reg) ;
|
||
|
|
||
|
if (mode == OUTPUT)
|
||
|
old &= (~mask) ;
|
||
|
else
|
||
|
old |= mask ;
|
||
|
|
||
|
wiringPiI2CWriteReg8 (node->fd, reg, old) ;
|
||
|
}
|
||
|
|
||
|
|
||
|
/*
|
||
|
* myDigitalWrite:
|
||
|
*********************************************************************************
|
||
|
*/
|
||
|
|
||
|
static void myDigitalWrite (struct wiringPiNodeStruct *node, int pin, int value)
|
||
|
{
|
||
|
int bit, old ;
|
||
|
|
||
|
pin -= node->pinBase ; // Pin now 0-15
|
||
|
|
||
|
bit = 1 << (pin & 7) ;
|
||
|
|
||
|
if (pin < 8) // Bank A
|
||
|
{
|
||
|
old = node->data2 ;
|
||
|
|
||
|
if (value == LOW)
|
||
|
old &= (~bit) ;
|
||
|
else
|
||
|
old |= bit ;
|
||
|
|
||
|
wiringPiI2CWriteReg8 (node->fd, MCP23016_GP0, old) ;
|
||
|
node->data2 = old ;
|
||
|
}
|
||
|
else // Bank B
|
||
|
{
|
||
|
old = node->data3 ;
|
||
|
|
||
|
if (value == LOW)
|
||
|
old &= (~bit) ;
|
||
|
else
|
||
|
old |= bit ;
|
||
|
|
||
|
wiringPiI2CWriteReg8 (node->fd, MCP23016_GP1, old) ;
|
||
|
node->data3 = old ;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
/*
|
||
|
* myDigitalRead:
|
||
|
*********************************************************************************
|
||
|
*/
|
||
|
|
||
|
static int myDigitalRead (struct wiringPiNodeStruct *node, int pin)
|
||
|
{
|
||
|
int mask, value, gpio ;
|
||
|
|
||
|
pin -= node->pinBase ;
|
||
|
|
||
|
if (pin < 8) // Bank A
|
||
|
gpio = MCP23016_GP0 ;
|
||
|
else
|
||
|
{
|
||
|
gpio = MCP23016_GP1 ;
|
||
|
pin &= 0x07 ;
|
||
|
}
|
||
|
|
||
|
mask = 1 << pin ;
|
||
|
value = wiringPiI2CReadReg8 (node->fd, gpio) ;
|
||
|
|
||
|
if ((value & mask) == 0)
|
||
|
return LOW ;
|
||
|
else
|
||
|
return HIGH ;
|
||
|
}
|
||
|
|
||
|
|
||
|
/*
|
||
|
* mcp23016Setup:
|
||
|
* Create a new instance of an MCP23016 I2C GPIO interface. We know it
|
||
|
* has 16 pins, so all we need to know here is the I2C address and the
|
||
|
* user-defined pin base.
|
||
|
*********************************************************************************
|
||
|
*/
|
||
|
|
||
|
int mcp23016Setup (const int pinBase, const int i2cAddress)
|
||
|
{
|
||
|
int fd ;
|
||
|
struct wiringPiNodeStruct *node ;
|
||
|
|
||
|
if ((fd = wiringPiI2CSetup (i2cAddress)) < 0)
|
||
|
return FALSE ;
|
||
|
|
||
|
wiringPiI2CWriteReg8 (fd, MCP23016_IOCON0, IOCON_INIT) ;
|
||
|
wiringPiI2CWriteReg8 (fd, MCP23016_IOCON1, IOCON_INIT) ;
|
||
|
|
||
|
node = wiringPiNewNode (pinBase, 16) ;
|
||
|
|
||
|
node->fd = fd ;
|
||
|
node->pinMode = myPinMode ;
|
||
|
node->digitalRead = myDigitalRead ;
|
||
|
node->digitalWrite = myDigitalWrite ;
|
||
|
node->data2 = wiringPiI2CReadReg8 (fd, MCP23016_OLAT0) ;
|
||
|
node->data3 = wiringPiI2CReadReg8 (fd, MCP23016_OLAT1) ;
|
||
|
|
||
|
return TRUE ;
|
||
|
}
|