| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406 |
- ##############################################
- # $Id: 52_I2C_SHT3x.pm 11780 2016-07-10 19:12:34Z macs00 $
- #
- # 52_I2C_SHT3x.pm
- #
- # i2c sensor for humidity and temperature
- #
- # Basis for this module is "52_I2C_SHT21.pm" from klausw. I adapted the module so that it
- # is suitable for the SHT3x family from Sensirion / Switzerland. At moment the SHT3x family
- # consists of 3 sensors:
- # SHT30 (Low-Cost): sensor with less accuracy
- # SHT31 (Standard): sensor with good accuracy
- # SHT35 (High-End): sensor with best accuracy
- #
- # Via hardware pin configuration the sensor can be configured for 2 different i2c addresses.
- #
- #
- # If you have any questions, suggestions or like to report a failure, please feel free to cantact me:
- # FHEM Forum username: macs
- #
- ##############################################
- package main;
- use strict;
- use warnings;
- use constant {
- # 0x44 (default): ADDR (pin 2) connected to VSS (supply voltage)
- # 0x45 : ADDR (pin 2) connected to VDD (ground)
- SHT3x_I2C_ADDRESS => '0x44',
- };
- ##################################################
- # Forward declarations
- #
- sub I2C_SHT3x_Initialize($);
- sub I2C_SHT3x_Define($$);
- sub I2C_SHT3x_Attr(@);
- sub I2C_SHT3x_Poll($);
- sub I2C_SHT3x_Set($@);
- sub I2C_SHT3x_Undef($$);
- sub I2C_SHT3x_DbLog_splitFn($);
- my %sets = (
- 'readValues' => 1,
- );
- sub I2C_SHT3x_Initialize($) {
- my ($hash) = @_;
- $hash->{DefFn} = 'I2C_SHT3x_Define';
- $hash->{InitFn} = 'I2C_SHT3x_Init';
- $hash->{AttrFn} = 'I2C_SHT3x_Attr';
- $hash->{SetFn} = 'I2C_SHT3x_Set';
- $hash->{UndefFn} = 'I2C_SHT3x_Undef';
- $hash->{I2CRecFn} = 'I2C_SHT3x_I2CRec';
- $hash->{AttrList} = 'IODev do_not_notify:0,1 showtime:0,1 poll_interval:1,2,5,10,20,30 ' .
- 'roundHumidityDecimal:0,1,2 roundTemperatureDecimal:0,1,2 ' .
- $readingFnAttributes;
- $hash->{DbLog_splitFn} = "I2C_SHT3x_DbLog_splitFn";
- }
- sub I2C_SHT3x_Define($$) {
- my ($hash, $def) = @_;
- my @a = split('[ \t][ \t]*', $def);
-
- $hash->{STATE} = "defined";
- if ($main::init_done) {
- eval { I2C_SHT3x_Init( $hash, [ @a[ 2 .. scalar(@a) - 1 ] ] ); };
- return I2C_SHT3x_Catch($@) if $@;
- }
- return undef;
- }
- sub I2C_SHT3x_Init($$) {
- my ( $hash, $args ) = @_;
-
- my $name = $hash->{NAME};
- if (defined $args && int(@$args) > 1)
- {
- return "Define: Wrong syntax. Usage:\n" .
- "define <name> I2C_SHT3x [<i2caddress>]";
- }
-
- if (defined (my $address = shift @$args)) {
- $hash->{I2C_Address} = $address =~ /^0.*$/ ? oct($address) : $address;
- return "$name I2C Address not valid" unless (($hash->{I2C_Address} < 128) && ($hash->{I2C_Address} > 3));
- } else {
- $hash->{I2C_Address} = hex(SHT3x_I2C_ADDRESS);
- }
- my $msg = '';
- # create default attributes
- if (AttrVal($name, 'poll_interval', '?') eq '?') {
- $msg = CommandAttr(undef, $name . ' poll_interval 5');
- if ($msg) {
- Log3 ($hash, 1, $msg);
- return $msg;
- }
- }
- AssignIoPort($hash);
- $hash->{STATE} = 'Initialized';
- return undef;
- }
- sub I2C_SHT3x_Catch($) {
- my $exception = shift;
- if ($exception) {
- $exception =~ /^(.*)( at.*FHEM.*)$/;
- return $1;
- }
- return undef;
- }
- sub I2C_SHT3x_Attr (@) {# hier noch Werteueberpruefung einfuegen
- my ($command, $name, $attr, $val) = @_;
- my $hash = $defs{$name};
- my $msg = '';
- if ($command && $command eq "set" && $attr && $attr eq "IODev") {
- eval {
- if ($main::init_done and (!defined ($hash->{IODev}) or $hash->{IODev}->{NAME} ne $val)) {
- main::AssignIoPort($hash,$val);
- my @def = split (' ',$hash->{DEF});
- I2C_SHT3x_Init($hash,\@def) if (defined ($hash->{IODev}));
- }
- };
- return I2C_SHT3x_Catch($@) if $@;
- }
- if ($attr eq 'poll_interval') {
- if ($val > 0) {
- RemoveInternalTimer($hash);
- InternalTimer(gettimeofday() + 5, 'I2C_SHT3x_Poll', $hash, 0);
- } else {
- $msg = 'Wrong poll intervall defined. poll_interval must be a number > 0';
- }
- } elsif ($attr eq 'roundHumidityDecimal') {
- $msg = 'Wrong $attr defined. Use one of 0, 1, 2' if defined($val) && $val <= 0 && $val >= 2 ;
- } elsif ($attr eq 'roundTemperatureDecimal') {
- $msg = 'Wrong $attr defined. Use one of 0, 1, 2' if defined($val) && $val <= 0 && $val >= 2 ;
- }
- return ($msg) ? $msg : undef;
- }
- sub I2C_SHT3x_Poll($) {
- my ($hash) = @_;
- my $name = $hash->{NAME};
-
- # Read values
- I2C_SHT3x_Set($hash, ($name, 'readValues'));
-
- my $pollInterval = AttrVal($hash->{NAME}, 'poll_interval', 0);
- if ($pollInterval > 0) {
- InternalTimer(gettimeofday() + ($pollInterval * 60), 'I2C_SHT3x_Poll', $hash, 0);
- }
- }
- sub I2C_SHT3x_Set($@) {
- my ($hash, @a) = @_;
- my $name = $a[0];
- my $cmd = $a[1];
- if(!defined($sets{$cmd})) {
- return 'Unknown argument ' . $cmd . ', choose one of ' . join(' ', keys %sets)
- }
-
- if ($cmd eq 'readValues') {
- I2C_SHT3x_triggerTempHum($hash);
- }
- }
- sub I2C_SHT3x_Undef($$) {
- my ($hash, $arg) = @_;
- RemoveInternalTimer($hash);
- return undef;
- }
- sub I2C_SHT3x_I2CRec ($$) {
- my ($hash, $clientmsg) = @_;
- my $name = $hash->{NAME};
- my $phash = $hash->{IODev};
- my $pname = $phash->{NAME};
- while ( my ( $k, $v ) = each %$clientmsg ) { #erzeugen von Internals fuer alle Keys in $clientmsg die mit dem physical Namen beginnen
- $hash->{$k} = $v if $k =~ /^$pname/ ;
- }
-
- # i2c data received?
- if ( $clientmsg->{direction} && $clientmsg->{$pname . "_SENDSTAT"} && $clientmsg->{$pname . "_SENDSTAT"} eq "Ok" ) {
- if ( $clientmsg->{direction} eq "i2cread" && defined($clientmsg->{received}) ) {
- Log3 $hash, 5, "empfangen: $clientmsg->{received}";
- my @raw = split(" ",$clientmsg->{received});
- I2C_SHT3x_GetTempHum ($hash, $clientmsg->{received}) if $clientmsg->{nbyte} == 6;
- }
- }
- }
- sub I2C_SHT3x_GetTempHum ($$) {
- my ($hash, $rawdata) = @_;
- my @raw = split(" ",$rawdata);
- if ( defined (my $crc = I2C_SHT3x_CheckCrc(@raw[0..2])) ) { #CRC Test
- Log3 $hash, 2, "CRC error temperature data(Temp_MSB Temp_LSB Temp_Chechsum Hum_MSB Hum_LSB Hum_Chechsum): $rawdata, Checksum calculated: $crc";
- $hash->{CRCErrorTemperature}++;
- return;
- }
- if ( defined (my $crc = I2C_SHT3x_CheckCrc(@raw[3..5])) ) { #CRC Test
- Log3 $hash, 2, "CRC error humidity data(Temp_MSB Temp_LSB Temp_Chechsum Hum_MSB Hum_LSB Hum_Chechsum): $rawdata, Checksum calculated: $crc";
- $hash->{CRCErrorHumidity}++;
- return;
- }
- my $name = $hash->{NAME};
- my $temperature = ($raw[0] << 8) | $raw[1];
- $temperature = ( 175.0 * ($temperature / ((2**16)-1.0) )) - 45.0;
- $temperature = sprintf(
- '%.' . AttrVal($hash->{NAME}, 'roundTemperatureDecimal', 1) . 'f',
- $temperature
- );
- my $humidity = ($raw[3] << 8) | $raw[4];
- $humidity = 100.0 * ($humidity / ((2**16)-1.0) );
- $humidity = sprintf(
- '%.' . AttrVal($hash->{NAME}, 'roundHumidityDecimal', 1) . 'f',
- $humidity
- );
- readingsBeginUpdate($hash);
- readingsBulkUpdate(
- $hash,
- 'state',
- 'T: ' . $temperature . ' H: ' . $humidity
- );
- readingsBulkUpdate($hash, 'temperature', $temperature);
- readingsBulkUpdate($hash, 'humidity', $humidity);
- readingsEndUpdate($hash, 1);
- }
- sub I2C_SHT3x_triggerTempHum($) {
- my ($hash) = @_;
- my $name = $hash->{NAME};
- return "$name: no IO device defined" unless ($hash->{IODev});
- my $phash = $hash->{IODev};
- my $pname = $phash->{NAME};
-
- # Write decimal 36 00 to device. This requests a "Repeatability = High, Clock stretching = disabled" temperature and humidity reading
- my $i2creq = { i2caddress => $hash->{I2C_Address}, direction => "i2cwrite" };
- $i2creq->{data} = "36 00";
- CallFn($pname, "I2CWrtFn", $phash, $i2creq);
- RemoveInternalTimer($hash);
- InternalTimer(gettimeofday() + 1, 'I2C_SHT3x_readValue', $hash, 0); #nach 1s Wert lesen
- return;
- }
- sub I2C_SHT3x_readValue($) {
- my ($hash) = @_;
- my $name = $hash->{NAME};
- return "$name: no IO device defined" unless ($hash->{IODev});
- my $phash = $hash->{IODev};
- my $pname = $phash->{NAME};
-
- # Reset Internal Timer to Poll Sub
- RemoveInternalTimer($hash);
- my $pollInterval = AttrVal($hash->{NAME}, 'poll_interval', 0);
- InternalTimer(gettimeofday() + ($pollInterval * 60), 'I2C_SHT3x_Poll', $hash, 0) if ($pollInterval > 0);
- # Read 6 byte: Temp MSB, Temp LSB, CRC, Hum MSB, Hum LSB, CRC
- my $i2cread = { i2caddress => $hash->{I2C_Address}, direction => "i2cread" };
- $i2cread->{nbyte} = 6;
- CallFn($pname, "I2CWrtFn", $phash, $i2cread);
-
- return;
- }
- sub I2C_SHT3x_CheckCrc(@) {
- my @data = @_;
- my $crc = 0xFF;
- my $poly = 0x131; #P(x)=x^8+x^5+x^4+1 = 100110001
- for (my $n = 0; $n < (scalar(@data) - 1); ++$n) {
- $crc ^= $data[$n];
- for (my $bit = 8; $bit > 0; --$bit) {
- $crc = ($crc & 0x80 ? $poly : 0 ) ^ ($crc << 1);
- }
- }
- return ($crc == $data[scalar(@data)-1] ? undef : $crc);
- }
- sub I2C_SHT3x_DbLog_splitFn($) {
- my ($event) = @_;
- Log3 undef, 5, "in DbLog_splitFn empfangen: $event";
- my ($reading, $value, $unit) = "";
- my @parts = split(/ /,$event);
- $reading = shift @parts;
- $reading =~ tr/://d;
- $value = $parts[0];
- $unit = "\xB0C" if(lc($reading) =~ m/temp/);
- $unit = "%" if(lc($reading) =~ m/humi/);
- return ($reading, $value, $unit);
- }
- 1;
- =pod
- =begin html
- <a name="I2C_SHT3x"></a>
- <h3>I2C_SHT3x</h3>
- (en | <a href="commandref_DE.html#I2C_SHT3x">de</a>)
- <ul>
- <a name="I2C_SHT3x"></a>
- Provides an interface to the SHT30/SHT31 I2C Humidity sensor from <a href="http:\\www.sensirion.com">Sensirion</a>.
- The I2C messages are sent through an I2C interface module like <a href="#RPII2C">RPII2C</a>, <a href="#FRM">FRM</a>
- or <a href="#NetzerI2C">NetzerI2C</a> so this device must be defined first.<br>
- <b>attribute IODev must be set</b><br>
- <a name="I2C_SHT3xDefine"></a><br>
- <b>Define</b>
- <ul>
- <code>define <name> I2C_SHT3x [<I2C Address>]</code><br>
- where <code><I2C Address></code> is an 2 digit hexadecimal value:<br>
- ADDR (pin 2) connected to VSS (supply voltage): 0x44 (default, if <code><I2C Address></code> is not set)<br>
- ADDR (pin 2) connected to VDD (ground): 0x45<br>
- For compatible sensors also other values than 0x44 or 0x45 can be set.<br>
- <br>
- </ul>
- <a name="I2C_SHT3xSet"></a>
- <b>Set</b>
- <ul>
- <code>set <name> readValues</code><br>
- Reads the current temperature and humidity values from sensor.<br><br>
- </ul>
- <a name="I2C_SHT3xAttr"></a>
- <b>Attributes</b>
- <ul>
- <li>poll_interval<br>
- Set the polling interval in minutes to query data from sensor<br>
- Default: 5, valid values: 1,2,5,10,20,30<br><br>
- </li>
- <li>roundHumidityDecimal, roundTemperatureDecimal<br>
- Number of decimal places for humidity or temperature value<br>
- Default: 1, valid values: 0 1 2<br><br>
- </li>
- <li><a href="#IODev">IODev</a></li>
- <li><a href="#do_not_notify">do_not_notify</a></li>
- <li><a href="#showtime">showtime</a></li>
- </ul><br>
- </ul>
- =end html
- =begin html_DE
- <a name="I2C_SHT3x"></a>
- <h3>I2C_SHT3x</h3>
- (<a href="commandref.html#I2C_SHT3x">en</a> | de)
- <ul>
- <a name="I2C_SHT3x"></a>
- Ermöglicht die Verwendung eines SHT30/SHT31 I2C Feuchtesensors von <a href="http:\\www.sensirion.com">Sensirion</a>.
- I2C-Botschaften werden über ein I2C Interface Modul wie beispielsweise das <a href="#RPII2C">RPII2C</a>, <a href="#FRM">FRM</a>
- oder <a href="#NetzerI2C">NetzerI2C</a> gesendet. Daher muss dieses vorher definiert werden.<br>
- <b>Das Attribut IODev muss definiert sein.</b><br>
- <a name="I2C_SHT3xDefine"></a><br>
- <b>Define</b>
- <ul>
- <code>define <name> I2C_SHT3x [<I2C Address>]</code><br>
- <br>
- Der Wert <code><I2C Address></code> ist ein zweistelliger Hex-Wert:<br>
- ADDR (Pin 2) verbunden mit VSS (Versorgungsspannung): 0x44 (Standardwert, wenn <code><I2C Address></code> nicht angegeben)<br>
- ADDR (pin 2) verbunden mit VDD (Masse): 0x45<br>
- Für kompatible Sensoren können auch andere Werte als 0x44 oder 0x45 angegeben werden.<br>
- <br>
- </ul>
- <a name="I2C_SHT3xSet"></a>
- <b>Set</b>
- <ul>
- <code>set <name> readValues</code><br>
- Aktuelle Temperatur und Feuchte Werte vom Sensor lesen.<br><br>
- </ul>
- <a name="I2C_SHT3xAttr"></a>
- <b>Attribute</b>
- <ul>
- <li>poll_interval<br>
- Aktualisierungsintervall aller Werte in Minuten.<br>
- Standard: 5, gültige Werte: 1,2,5,10,20,30<br><br>
- </li>
- <li>roundHumidityDecimal, roundTemperatureDecimal<br>
- Anzahl Dezimalstellen für den Feuchte- oder Temperaturwert<br>
- Standard: 1, gültige Werte: 0 1 2<br><br>
- </li>
- <li><a href="#IODev">IODev</a></li>
- <li><a href="#do_not_notify">do_not_notify</a></li>
- <li><a href="#showtime">showtime</a></li>
- </ul><br>
- </ul>
- =end html_DE
- =cut
|