You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
188 lines
4.5 KiB
188 lines
4.5 KiB
/* |
|
* mcp23s08.c: |
|
* Extend wiringPi with the MCP 23s08 SPI 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 <stdint.h> |
|
|
|
#include "wiringPi.h" |
|
#include "wiringPiSPI.h" |
|
#include "mcp23x0817.h" |
|
|
|
#include "mcp23s08.h" |
|
|
|
#define MCP_SPEED 4000000 |
|
|
|
|
|
|
|
/* |
|
* writeByte: |
|
* Write a byte to a register on the MCP23s08 on the SPI bus. |
|
********************************************************************************* |
|
*/ |
|
|
|
static void writeByte (uint8_t spiPort, uint8_t devId, uint8_t reg, uint8_t data) |
|
{ |
|
uint8_t spiData [4] ; |
|
|
|
spiData [0] = CMD_WRITE | ((devId & 7) << 1) ; |
|
spiData [1] = reg ; |
|
spiData [2] = data ; |
|
|
|
wiringPiSPIDataRW (spiPort, spiData, 3) ; |
|
} |
|
|
|
/* |
|
* readByte: |
|
* Read a byte from a register on the MCP23s08 on the SPI bus. |
|
********************************************************************************* |
|
*/ |
|
|
|
static uint8_t readByte (uint8_t spiPort, uint8_t devId, uint8_t reg) |
|
{ |
|
uint8_t spiData [4] ; |
|
|
|
spiData [0] = CMD_READ | ((devId & 7) << 1) ; |
|
spiData [1] = reg ; |
|
|
|
wiringPiSPIDataRW (spiPort, spiData, 3) ; |
|
|
|
return spiData [2] ; |
|
} |
|
|
|
|
|
/* |
|
* myPinMode: |
|
********************************************************************************* |
|
*/ |
|
|
|
static void myPinMode (struct wiringPiNodeStruct *node, int pin, int mode) |
|
{ |
|
int mask, old, reg ; |
|
|
|
reg = MCP23x08_IODIR ; |
|
mask = 1 << (pin - node->pinBase) ; |
|
old = readByte (node->data0, node->data1, reg) ; |
|
|
|
if (mode == OUTPUT) |
|
old &= (~mask) ; |
|
else |
|
old |= mask ; |
|
|
|
writeByte (node->data0, node->data1, reg, old) ; |
|
} |
|
|
|
|
|
/* |
|
* myPullUpDnControl: |
|
********************************************************************************* |
|
*/ |
|
|
|
static void myPullUpDnControl (struct wiringPiNodeStruct *node, int pin, int mode) |
|
{ |
|
int mask, old, reg ; |
|
|
|
reg = MCP23x08_GPPU ; |
|
mask = 1 << (pin - node->pinBase) ; |
|
|
|
old = readByte (node->data0, node->data1, reg) ; |
|
|
|
if (mode == PUD_UP) |
|
old |= mask ; |
|
else |
|
old &= (~mask) ; |
|
|
|
writeByte (node->data0, node->data1, reg, old) ; |
|
} |
|
|
|
|
|
/* |
|
* myDigitalWrite: |
|
********************************************************************************* |
|
*/ |
|
|
|
static void myDigitalWrite (struct wiringPiNodeStruct *node, int pin, int value) |
|
{ |
|
int bit, old ; |
|
|
|
bit = 1 << ((pin - node->pinBase) & 7) ; |
|
|
|
old = node->data2 ; |
|
if (value == LOW) |
|
old &= (~bit) ; |
|
else |
|
old |= bit ; |
|
|
|
writeByte (node->data0, node->data1, MCP23x08_GPIO, old) ; |
|
node->data2 = old ; |
|
} |
|
|
|
|
|
/* |
|
* myDigitalRead: |
|
********************************************************************************* |
|
*/ |
|
|
|
static int myDigitalRead (struct wiringPiNodeStruct *node, int pin) |
|
{ |
|
int mask, value ; |
|
|
|
mask = 1 << ((pin - node->pinBase) & 7) ; |
|
value = readByte (node->data0, node->data1, MCP23x08_GPIO) ; |
|
|
|
if ((value & mask) == 0) |
|
return LOW ; |
|
else |
|
return HIGH ; |
|
} |
|
|
|
|
|
/* |
|
* mcp23s08Setup: |
|
* Create a new instance of an MCP23s08 SPI GPIO interface. We know it |
|
* has 8 pins, so all we need to know here is the SPI address and the |
|
* user-defined pin base. |
|
********************************************************************************* |
|
*/ |
|
|
|
int mcp23s08Setup (const int pinBase, const int spiPort, const int devId) |
|
{ |
|
struct wiringPiNodeStruct *node ; |
|
|
|
if (wiringPiSPISetup (spiPort, MCP_SPEED) < 0) |
|
return FALSE ; |
|
|
|
writeByte (spiPort, devId, MCP23x08_IOCON, IOCON_INIT) ; |
|
|
|
node = wiringPiNewNode (pinBase, 8) ; |
|
|
|
node->data0 = spiPort ; |
|
node->data1 = devId ; |
|
node->pinMode = myPinMode ; |
|
node->pullUpDnControl = myPullUpDnControl ; |
|
node->digitalRead = myDigitalRead ; |
|
node->digitalWrite = myDigitalWrite ; |
|
node->data2 = readByte (spiPort, devId, MCP23x08_OLAT) ; |
|
|
|
return TRUE ; |
|
}
|
|
|