Library to read and write Azoteq IQS6xx device registers via I2C.

Dependents:   IQS620_HelloWorld IQS622_HelloWorld IQS624_HelloWorld IQS621_HelloWorld ... more

Library: IQS62x

Library to read and write Azoteq IQS6xx device registers via I2C.

Supported Devices

Components / IQS620A
Ultra low power sensor for magnetic field, capacitive touch and inductive proximity. Empowers next-generation user interfaces.

Components / IQS621
Azoteq IQS621 ultra low power sensor for ambient light, magnetic field, capacitance and inductive proximity. Empowers next-generation user interfaces.

Components / IQS622
Azoteq IQS622 ultra low power sensor for ambient light, active (reflective) IR, magnetic field, capacitance and inductive proximity. Empowers next-generation user interfaces.

Components / IQS624
Ultra low power sensor for rotating magnetic field, capacitive touch, and inductive proximity. Empowers next-generation user interfaces.

Handy Table of ProxFusion Device Features


ALS = Ambient Light Sensor PIR = Passive Infrared

/media/uploads/AzqDev/mbed-azoteq-proxfusion-handy-table-of-product-features.jpg

Committer:
AzqDev
Date:
Thu May 18 15:23:40 2017 +0000
Revision:
19:f0675a3150c7
Parent:
18:b85048ec91d2
Child:
20:7eb08a52b954
comments

Who changed what in which revision?

UserRevisionLine numberNew contents of line
AzqDev 0:ce75ae1e8fc7 1 // A class library for Azoteq IQS62x devices
AzqDev 3:e26d7c502309 2
AzqDev 2:c16cb655d4a4 3 // Copyright 2017 Azoteq. 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.
AzqDev 2:c16cb655d4a4 4
AzqDev 17:2d46eabb129e 5 // More info on IQS62x sensor ICs: http://www.azoteq.com/products/proxfusion?mbed
AzqDev 3:e26d7c502309 6
AzqDev 18:b85048ec91d2 7 // ProxFusion 5-minute YouTube video: http://bit.ly/proxfusion-video
AzqDev 5:653138b5cae9 8
AzqDev 0:ce75ae1e8fc7 9 #include "IQS62x.h"
AzqDev 0:ce75ae1e8fc7 10
AzqDev 0:ce75ae1e8fc7 11 // constructor
AzqDev 12:5a9bbbd6e312 12 IQS62xIO::IQS62xIO() :
AzqDev 0:ce75ae1e8fc7 13 i2c( IQS_I2C_DATA_PIN, IQS_I2C_CLOCK_PIN ), // first run the constructor for mbed class I2C
AzqDev 0:ce75ae1e8fc7 14 IQSready( IQS_READY_PIN ) // first run the constructor for mbed class DigitalIn
AzqDev 12:5a9bbbd6e312 15 {
AzqDev 0:ce75ae1e8fc7 16 registers = I2CBuffer; // pointer to the receive buffer
AzqDev 0:ce75ae1e8fc7 17 I2CErrorCount = 0; // reset I2C error count
AzqDev 14:2514595e2753 18 IQSframes = 0; // reset number of reads
AzqDev 0:ce75ae1e8fc7 19 i2c.frequency( I2Cspeed ); // I2C clock frequency
AzqDev 12:5a9bbbd6e312 20
AzqDev 12:5a9bbbd6e312 21 // this uses memory but is very handy in diagnostics
AzqDev 14:2514595e2753 22 memset(writeFlag, 0, I2CBufferSize); // a table to remmber if we wrote to a register
AzqDev 14:2514595e2753 23 memset(lastWrite, 0, I2CBufferSize); // a table to remember what we wrote to a register
AzqDev 14:2514595e2753 24 memset(lastRead, 0, I2CBufferSize); // a table to remmber what we read the previous read cycle
AzqDev 14:2514595e2753 25 memset(readChanges, 0, I2CBufferSize); // a table to mark up any registers that changed since the previous read
AzqDev 14:2514595e2753 26 memset(readChangesEver, 0, I2CBufferSize); // a table to mark up any registers that changed BUT never forget
AzqDev 19:f0675a3150c7 27 memset(writeChanges, 0, I2CBufferSize); // a table to mark up any registers that differ from what was written to it
AzqDev 12:5a9bbbd6e312 28 }
AzqDev 0:ce75ae1e8fc7 29
AzqDev 7:5e7ea07265d9 30 // write a single byte to an IQS62x register
AzqDev 12:5a9bbbd6e312 31 void IQS62xIO::writeRegister(int address, int data)
AzqDev 12:5a9bbbd6e312 32 {
AzqDev 12:5a9bbbd6e312 33 writeFlag[ address & 0xff ] = 1; // remember which registers we changed
AzqDev 12:5a9bbbd6e312 34 lastWrite[ address & 0xff ] = data & 0xff; // remember what we wrote
AzqDev 7:5e7ea07265d9 35 char twoBytes [2];
AzqDev 7:5e7ea07265d9 36 int numberOfBytes = 2;
AzqDev 7:5e7ea07265d9 37 twoBytes[0] = address & 0xff;
AzqDev 7:5e7ea07265d9 38 twoBytes[1] = data & 0xff;
AzqDev 12:5a9bbbd6e312 39 waitForIqsReady();
AzqDev 12:5a9bbbd6e312 40 if(0!=i2c.write(I2C_ADR,twoBytes,numberOfBytes,false))
AzqDev 0:ce75ae1e8fc7 41 I2CErrorCount++;
AzqDev 0:ce75ae1e8fc7 42 }
AzqDev 0:ce75ae1e8fc7 43
AzqDev 7:5e7ea07265d9 44 // configure the IQS62x
AzqDev 8:f9c71b39c602 45 #include "deviceType.h" // to check if we should override the configuration definition
AzqDev 8:f9c71b39c602 46 #ifndef OVERRIDE_CONFIGURE
AzqDev 9:7c2666dfbc9a 47 // if there is no override for configure() then we use this simple version
AzqDev 12:5a9bbbd6e312 48 void IQS62xIO::configure()
AzqDev 12:5a9bbbd6e312 49 {
AzqDev 7:5e7ea07265d9 50 writeRegister(0xd0,0x40); // simplest config : just acknowledge/clear the reset flag
AzqDev 7:5e7ea07265d9 51 }
AzqDev 8:f9c71b39c602 52 #endif
AzqDev 7:5e7ea07265d9 53
AzqDev 10:8fdac52881aa 54 // read N registers, starting at provided offset
AzqDev 12:5a9bbbd6e312 55 void IQS62xIO::readIqsRegisters(int start, int count)
AzqDev 12:5a9bbbd6e312 56 {
AzqDev 10:8fdac52881aa 57 memset(I2CBuffer,0x55,I2CBufferSize); // "clear" i2c receive buffer
AzqDev 14:2514595e2753 58 IQSframes++; // count frames
AzqDev 12:5a9bbbd6e312 59 waitForIqsReady();
AzqDev 12:5a9bbbd6e312 60 char i2c_start_address [1];
AzqDev 10:8fdac52881aa 61 i2c_start_address[0] = start & 0xff;
AzqDev 10:8fdac52881aa 62 int numberOfBytes = 1;
AzqDev 10:8fdac52881aa 63 // write start address to the IQS62x address register
AzqDev 12:5a9bbbd6e312 64 if(0!=i2c.write(I2C_ADR,i2c_start_address,numberOfBytes,false))
AzqDev 0:ce75ae1e8fc7 65 I2CErrorCount++;
AzqDev 12:5a9bbbd6e312 66 waitForIqsReady();
AzqDev 10:8fdac52881aa 67 numberOfBytes = count % I2CBufferSize;
AzqDev 0:ce75ae1e8fc7 68 // read register values into a buffer
AzqDev 0:ce75ae1e8fc7 69 if(0!=i2c.read(I2C_ADR,I2CBuffer,numberOfBytes,false))
AzqDev 0:ce75ae1e8fc7 70 I2CErrorCount++;
AzqDev 12:5a9bbbd6e312 71
AzqDev 12:5a9bbbd6e312 72 #define PLEASE_CHECK_FOR_IQS_CHANGES 1
AzqDev 12:5a9bbbd6e312 73 #ifndef DONT_CHECK_FOR_IQS_CHANGES
AzqDev 12:5a9bbbd6e312 74 // this is optional but handy diagnostics
AzqDev 19:f0675a3150c7 75 // we build six tables:
AzqDev 19:f0675a3150c7 76 //writeFlag - a table to remmber if we wrote to a register
AzqDev 19:f0675a3150c7 77 //lastWrite - a table to remember what we wrote to a register
AzqDev 19:f0675a3150c7 78 //lastRead - a table to remmber what we read the previous read cycle
AzqDev 19:f0675a3150c7 79 //readChanges - a table to mark up any registers that changed since the previous read
AzqDev 19:f0675a3150c7 80 //readChangesEver - a table to mark up any registers that changed BUT never forget
AzqDev 19:f0675a3150c7 81 //writeChanges - a table to mark up any registers that differ from what was written to it
AzqDev 19:f0675a3150c7 82 if (IQSframes == 1) memcpy(lastRead, I2CBuffer,I2CBufferSize); // if this is the first frame init the history buffer
AzqDev 14:2514595e2753 83 if (IQSframes < 20) memset(readChangesEver,0,I2CBufferSize); // clear this until we have had 20 full reads
AzqDev 12:5a9bbbd6e312 84 for(int i=0; i<I2CBufferSize; i++) {
AzqDev 12:5a9bbbd6e312 85 if (writeFlag[i] == 1) { // only if we previously wrote to this register we check it
AzqDev 12:5a9bbbd6e312 86 if( I2CBuffer[i] != lastWrite[i] )
AzqDev 12:5a9bbbd6e312 87 writeChanges[i]=1; // if register different from what we wrote, mark it
AzqDev 12:5a9bbbd6e312 88 }
AzqDev 14:2514595e2753 89 if ( I2CBuffer[i] != lastRead[i] ) {
AzqDev 13:71f8ee16a3a1 90 readChanges[i]=20; // if register differs from previous read, highlight it for 20 read cycles
AzqDev 14:2514595e2753 91 if (readChangesEver[i] < 127) readChangesEver[i]++; // count changes up to 127 and never forget
AzqDev 14:2514595e2753 92 } else {
AzqDev 13:71f8ee16a3a1 93 if (readChanges[i] > 0) readChanges[i]--; // bring it closer to zero
AzqDev 13:71f8ee16a3a1 94 }
AzqDev 12:5a9bbbd6e312 95 }
AzqDev 19:f0675a3150c7 96 memcpy(lastRead, I2CBuffer, I2CBufferSize); // preserve history for next round
AzqDev 12:5a9bbbd6e312 97 #endif
AzqDev 0:ce75ae1e8fc7 98 }
AzqDev 0:ce75ae1e8fc7 99
AzqDev 0:ce75ae1e8fc7 100 // wait for IQS62x to provide a ready signal (low) on IQS62x_ready pin
AzqDev 12:5a9bbbd6e312 101 void IQS62xIO::waitForIqsReady()
AzqDev 12:5a9bbbd6e312 102 {
AzqDev 12:5a9bbbd6e312 103 int timeout=0;
AzqDev 0:ce75ae1e8fc7 104 while (1) {
AzqDev 0:ce75ae1e8fc7 105 if(IQSready==1) break;
AzqDev 12:5a9bbbd6e312 106 if (timeout++ > 1000000) goto fatal_error;
AzqDev 0:ce75ae1e8fc7 107 }
AzqDev 0:ce75ae1e8fc7 108 timeout=0;
AzqDev 0:ce75ae1e8fc7 109 while (1) {
AzqDev 0:ce75ae1e8fc7 110 if(IQSready==0) break;
AzqDev 0:ce75ae1e8fc7 111 if (timeout++ > 1000000) goto fatal_error;
AzqDev 0:ce75ae1e8fc7 112 }
AzqDev 16:b77c819f6c6a 113 return;;
AzqDev 12:5a9bbbd6e312 114 fatal_error:
AzqDev 12:5a9bbbd6e312 115 error ("Fatal Error: IQS62x ready pin is not toggling");
AzqDev 16:b77c819f6c6a 116 }
AzqDev 16:b77c819f6c6a 117
AzqDev 16:b77c819f6c6a 118 // return one of the tables of changes that we maintain
AzqDev 16:b77c819f6c6a 119 char * IQS62xIO::getTable( int option )
AzqDev 16:b77c819f6c6a 120 {
AzqDev 16:b77c819f6c6a 121 switch ( option ) {
AzqDev 16:b77c819f6c6a 122 case 1 : return readChangesEver; // a table to flag any register that ever changed
AzqDev 16:b77c819f6c6a 123 case 2 : return readChanges; // a table to flag any register that changed in the last 20 reads
AzqDev 16:b77c819f6c6a 124 case 3 : return writeFlag; // a table to flag any register we wrote to or initialized
AzqDev 16:b77c819f6c6a 125 case 4 : return writeChanges; // a table to flag any register we wrote to but then its value changed
AzqDev 16:b77c819f6c6a 126 default: return NULL;;
AzqDev 16:b77c819f6c6a 127 }
AzqDev 0:ce75ae1e8fc7 128 }