// // FILE: AM2315C.cpp // AUTHOR: Rob Tillaart // VERSION: 0.1.0 // PURPOSE: Arduino library for AM2315C I2C temperature and humidity sensor. #include "AM2315C.h" // set AM2315C_WIRE_TIME_OUT to 0 to disable. // note this timeout is commented in code below. #define AM2315C_WIRE_TIME_OUT 250000 // microseconds const uint8_t AM2315C_ADDRESS = 0x38; AM2315C::AM2315C(TwoWire *wire) { _wire = wire; // reset() ? _temperature = 0; _humidity = 0; _humOffset = 0; _tempOffset = 0; _status = AM2315C_OK; _lastRequest = 0; _lastRead = 0; } bool AM2315C::begin() { _wire->begin(); // _wire->setWireTimeout(AM2315C_WIRE_TIME_OUT, true); return isConnected(); } #if defined(ESP8266) || defined(ESP32) bool AM2315C::begin(const uint8_t dataPin, const uint8_t clockPin) { if ((dataPin < 255) && (clockPin < 255)) { _wire->begin(dataPin, clockPin); } else { _wire->begin(); } // _wire->setWireTimeout(AM2315C_WIRE_TIME_OUT, true); return isConnected(); } #endif bool AM2315C::isConnected() { _wire->beginTransmission(AM2315C_ADDRESS); int rv = _wire->endTransmission(); return rv == 0; } uint8_t AM2315C::getAddress() { return AM2315C_ADDRESS; } // See datasheet 7.4 Sensor Reading Process, point 1 // use with care. uint8_t AM2315C::resetSensor() { uint8_t count = 255; if ((readStatus() & 0x18) != 0x18) { count++; if (_resetRegister(0x1B)) count++; if (_resetRegister(0x1C)) count++; if (_resetRegister(0x1E)) count++; delay(10); } return count; } //////////////////////////////////////////////// // // READ THE SENSOR // int AM2315C::read() { // do not read to fast == more than once per second. if (millis() - _lastRead < 1000) { return AM2315C_ERROR_LASTREAD; } int status = requestData(); if (status < 0) return status; // wait for measurement ready while (isMeasuring()) { yield(); } // read the measurement status = readData(); if (status < 0) return status; // convert it to meaningful data return convert(); } int AM2315C::requestData() { // reset sensor if needed. resetSensor(); // GET CONNECTION _wire->beginTransmission(AM2315C_ADDRESS); _wire->write(0xAC); _wire->write(0x33); _wire->write(0x00); int rv = _wire->endTransmission(); _lastRequest = millis(); return rv; } int AM2315C::readData() { // GET DATA const uint8_t length = 7; int bytes = _wire->requestFrom(AM2315C_ADDRESS, length); if (bytes == 0) return AM2315C_ERROR_CONNECT; if (bytes < length) return AM2315C_MISSING_BYTES; bool allZero = true; for (int i = 0; i < bytes; i++) { _bits[i] = _wire->read(); // if (_bits[i] < 0x10) Serial.print(0); // Serial.print(_bits[i], HEX); // Serial.print(" "); allZero = allZero && (_bits[i] == 0); } // Serial.println(); if (allZero) return AM2315C_ERROR_BYTES_ALL_ZERO; _lastRead = millis(); return bytes; } int AM2315C::convert() { // CONVERT AND STORE _status = _bits[0]; uint32_t raw = _bits[1]; raw <<= 8; raw += _bits[2]; raw <<= 4; raw += (_bits[3] >> 4); _humidity = raw * 9.5367431640625e-5; // ==> / 1048576.0 * 100%; raw = (_bits[3] & 0x0F); raw <<= 8; raw += _bits[4]; raw <<= 8; raw += _bits[5]; _temperature = raw * 1.9073486328125e-4 - 50; // ==> / 1048576.0 * 200 - 50; // TEST CHECKSUM uint8_t _crc = _crc8(_bits, 6); // Serial.print(_crc, HEX); // Serial.print("\t"); // Serial.println(_bits[6], HEX); if (_crc != _bits[6]) return AM2315C_ERROR_CHECKSUM; return AM2315C_OK; } //////////////////////////////////////////////// // // TEMPERATURE & HUMIDITY & OFFSET // float AM2315C::getHumidity() { return _humidity + _humOffset; }; float AM2315C::getTemperature() { return _temperature + _tempOffset; }; void AM2315C::setHumOffset(float offset) { _humOffset = offset; }; void AM2315C::setTempOffset(float offset) { _tempOffset = offset; }; float AM2315C::getHumOffset() { return _humOffset; }; float AM2315C::getTempOffset() { return _tempOffset; }; //////////////////////////////////////////////// // // STATUS // uint8_t AM2315C::readStatus() { _wire->requestFrom(AM2315C_ADDRESS, (uint8_t)1); delay(1); // needed to stabilize timing return (uint8_t) _wire->read(); } bool AM2315C::isCalibrated() { return (readStatus() & 0x08) == 0x08; } bool AM2315C::isMeasuring() { return (readStatus() & 0x80) == 0x80; } bool AM2315C::isIdle() { return (readStatus() & 0x80) == 0x00; } int AM2315C::internalStatus() { return _status; }; //////////////////////////////////////////////// // // TIMING // uint32_t AM2315C::lastRead() { return _lastRead; }; uint32_t AM2315C::lastRequest() { return _lastRequest; }; //////////////////////////////////////////////// // // PRIVATE // uint8_t AM2315C::_crc8(uint8_t *ptr, uint8_t len) { uint8_t crc = 0xFF; while(len--) { crc ^= *ptr++; for (uint8_t i = 0; i < 8; i++) { if (crc & 0x80) { crc <<= 1; crc ^= 0x31; } else { crc <<= 1; } } } return crc; } // Code based on demo code sent by www.aosong.com // no further documentation. // 0x1B returned 18, 0, 4 // 0x1C returned 18, 65, 0 // 0x1E returned 18, 8, 0 // 18 seems to be status register // other values unknown. bool AM2315C::_resetRegister(uint8_t reg) { uint8_t value[3]; _wire->beginTransmission(AM2315C_ADDRESS); _wire->write(reg); _wire->write(0x00); _wire->write(0x00); if (_wire->endTransmission() != 0) return false; delay(5); int bytes = _wire->requestFrom(AM2315C_ADDRESS, (uint8_t)3); for (int i = 0; i < bytes; i++) { value[i] = _wire->read(); // Serial.println(value[i], HEX); } delay(10); _wire->beginTransmission(AM2315C_ADDRESS); _wire->write(0xB0 | reg); _wire->write(value[1]); _wire->write(value[2]); if (_wire->endTransmission() != 0) return false; delay(5); return true; } // -- END OF FILE --