/* * Copyright (c) 2007, Swedish Institute of Computer Science * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * 3. Neither the name of the Institute nor the names of its contributors * may be used to endorse or promote products derived from this software * without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE INSTITUTE AND CONTRIBUTORS ``AS IS'' AND * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE INSTITUTE OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. * */ /* * Device driver for the Sensirion SHT1x/SHT7x family of humidity and * temperature sensors. */ #include "contiki.h" #include <stdio.h> #include "dev/sht11/sht11.h" #include "sht11-arch.h" #define DEBUG 0 #if DEBUG #include <stdio.h> #define PRINTF(...) printf(__VA_ARGS__) #else #define PRINTF(...) #endif #ifndef SDA_0 #define SDA_0() (SHT11_PxDIR |= BV(SHT11_ARCH_SDA)) /* SDA Output=0 */ #define SDA_1() (SHT11_PxDIR &= ~BV(SHT11_ARCH_SDA)) /* SDA Input */ #define SDA_IS_1 (SHT11_PxIN & BV(SHT11_ARCH_SDA)) #define SCL_0() (SHT11_PxOUT &= ~BV(SHT11_ARCH_SCL)) /* SCL Output=0 */ #define SCL_1() (SHT11_PxOUT |= BV(SHT11_ARCH_SCL)) /* SCL Output=1 */ #endif /* adr command r/w */ #define STATUS_REG_W 0x06 /* 000 0011 0 */ #define STATUS_REG_R 0x07 /* 000 0011 1 */ #define MEASURE_TEMP 0x03 /* 000 0001 1 */ #define MEASURE_HUMI 0x05 /* 000 0010 1 */ #define RESET 0x1e /* 000 1111 0 */ /* This can probably be reduced to 250ns according to data sheet. */ #ifndef delay_400ns #define delay_400ns() _NOP() #endif /*---------------------------------------------------------------------------*/ static void sstart(void) { SDA_1(); SCL_0(); delay_400ns(); SCL_1(); delay_400ns(); SDA_0(); delay_400ns(); SCL_0(); delay_400ns(); SCL_1(); delay_400ns(); SDA_1(); delay_400ns(); SCL_0(); } /*---------------------------------------------------------------------------*/ static void sreset(void) { int i; SDA_1(); SCL_0(); for(i = 0; i < 9 ; i++) { SCL_1(); delay_400ns(); SCL_0(); delay_400ns(); } sstart(); /* Start transmission, why??? */ } /*---------------------------------------------------------------------------*/ /* * Return true if we received an ACK. */ static int swrite(unsigned _c) { unsigned char c = _c; int i; int ret; for(i = 0; i < 8; i++, c <<= 1) { if(c & 0x80) { SDA_1(); } else { SDA_0(); } SCL_1(); delay_400ns(); SCL_0(); delay_400ns(); } SDA_1(); SCL_1(); delay_400ns(); ret = !SDA_IS_1; SCL_0(); return ret; } /*---------------------------------------------------------------------------*/ static unsigned sread(int send_ack) { int i; unsigned char c = 0x00; SDA_1(); for(i = 0; i < 8; i++) { c <<= 1; SCL_1(); delay_400ns(); if(SDA_IS_1) { c |= 0x1; } SCL_0(); delay_400ns(); } if(send_ack) { SDA_0(); } SCL_1(); delay_400ns(); SCL_0(); SDA_1(); /* Release SDA */ return c; } /*---------------------------------------------------------------------------*/ #define CRC_CHECK #ifdef CRC_CHECK static unsigned char rev8bits(unsigned char v) { unsigned char r = v; int s = 7; for (v >>= 1; v; v >>= 1) { r <<= 1; r |= v & 1; s--; } r <<= s; /* Shift when v's highest bits are zero */ return r; } /*---------------------------------------------------------------------------*/ /* BEWARE: Bit reversed CRC8 using polynomial ^8 + ^5 + ^4 + 1 */ static unsigned crc8_add(unsigned acc, unsigned byte) { int i; acc ^= byte; for(i = 0; i < 8; i++) { if(acc & 0x80) { acc = (acc << 1) ^ 0x31; } else { acc <<= 1; } } return acc & 0xff; } #endif /* CRC_CHECK */ /*---------------------------------------------------------------------------*/ /* * Power up the device. The device can be used after an additional * 11ms waiting time. */ void sht11_init(void) { /* * SCL Output={0,1} * SDA 0: Output=0 * 1: Input and pull-up (Output=0) */ #ifdef SHT11_INIT SHT11_INIT(); #else /* As this driver is bit-bang based, disable the I2C first This assumes the SDA/SCL pins passed in the -arch.h file are actually the same used for I2C operation, else comment out the following */ #warning SHT11: DISABLING I2C BUS SHT11_PxSEL &= ~(BV(SHT11_ARCH_SDA) | BV(SHT11_ARCH_SCL)); #if defined(__MSP430_HAS_MSP430X_CPU__) || defined(__MSP430_HAS_MSP430XV2_CPU__) SHT11_PxREN &= ~(BV(SHT11_ARCH_SDA) | BV(SHT11_ARCH_SCL)); #endif /* Configure SDA/SCL as GPIOs */ SHT11_PxOUT |= BV(SHT11_ARCH_PWR); SHT11_PxOUT &= ~(BV(SHT11_ARCH_SDA) | BV(SHT11_ARCH_SCL)); SHT11_PxDIR |= BV(SHT11_ARCH_PWR) | BV(SHT11_ARCH_SCL); #endif } /*---------------------------------------------------------------------------*/ /* * Power of device. */ void sht11_off(void) { #ifdef SHT11_OFF SHT11_OFF(); #else SHT11_PxOUT &= ~BV(SHT11_ARCH_PWR); SHT11_PxOUT &= ~(BV(SHT11_ARCH_SDA) | BV(SHT11_ARCH_SCL)); SHT11_PxDIR |= BV(SHT11_ARCH_PWR) | BV(SHT11_ARCH_SCL); #endif } /*---------------------------------------------------------------------------*/ /* * Only commands MEASURE_HUMI or MEASURE_TEMP! */ static unsigned int scmd(unsigned cmd) { unsigned int n; if(cmd != MEASURE_HUMI && cmd != MEASURE_TEMP) { PRINTF("Illegal command: %d\n", cmd); return -1; } sstart(); /* Start transmission */ if(!swrite(cmd)) { PRINTF("SHT11: scmd - swrite failed\n"); goto fail; } for(n = 0; n < 20000; n++) { if(!SDA_IS_1) { unsigned t0, t1, rcrc; t0 = sread(1); t1 = sread(1); rcrc = sread(0); PRINTF("SHT11: scmd - read %d, %d\n", t0, t1); #ifdef CRC_CHECK { unsigned crc; crc = crc8_add(0x0, cmd); crc = crc8_add(crc, t0); crc = crc8_add(crc, t1); if(crc != rev8bits(rcrc)) { PRINTF("SHT11: scmd - crc check failed %d vs %d\n", crc, rev8bits(rcrc)); goto fail; } } #endif return (t0 << 8) | t1; } /* short wait before next loop */ clock_wait(1); } fail: sreset(); return -1; } /*---------------------------------------------------------------------------*/ /* * Call may take up to 210ms. */ unsigned int sht11_temp(void) { return scmd(MEASURE_TEMP); } /*---------------------------------------------------------------------------*/ /* * Call may take up to 210ms. */ unsigned int sht11_humidity(void) { return scmd(MEASURE_HUMI); } /*---------------------------------------------------------------------------*/ #if 1 /* But ok! */ unsigned sht11_sreg(void) { unsigned sreg, rcrc; sstart(); /* Start transmission */ if(!swrite(STATUS_REG_R)) { goto fail; } sreg = sread(1); rcrc = sread(0); #ifdef CRC_CHECK { unsigned crc; crc = crc8_add(0x0, STATUS_REG_R); crc = crc8_add(crc, sreg); if (crc != rev8bits(rcrc)) goto fail; } #endif return sreg; fail: sreset(); return -1; } #endif /*---------------------------------------------------------------------------*/ #if 0 int sht11_set_sreg(unsigned sreg) { sstart(); /* Start transmission */ if(!swrite(STATUS_REG_W)) { goto fail; } if(!swrite(sreg)) { goto fail; } return 0; fail: sreset(); return -1; } #endif /*---------------------------------------------------------------------------*/ #if 0 int sht11_reset(void) { sstart(); /* Start transmission */ if(!swrite(RESET)) { goto fail; } return 0; fail: sreset(); return -1; } #endif /*---------------------------------------------------------------------------*/