# HTU21D(F)/Si7013/Si7020/Si7021/SHT21 i2c based temperature sensors support
#
# Copyright (C) 2020  Lucio Tarantino <lucio.tarantino@gmail.com>
#
# This file may be distributed under the terms of the GNU GPLv3 license.
import logging
from . import bus

######################################################################
# NOTE: The implementation requires write support of length 0
#       before reading on the i2c bus of the mcu.
#
# Compatible Sensors:
#       HTU21D - Tested on Linux MCU.
#       Si7013 - Untested
#       Si7020 - Untested
#       Si7021 - Untested
#       SHT21  - Untested
#
######################################################################

HTU21D_I2C_ADDR= 0x40

HTU21D_COMMANDS = {
    'HTU21D_TEMP'       :0xE3,
    'HTU21D_HUMID'      :0xE5,
    'HTU21D_TEMP_NH'    :0xF3,
    'HTU21D_HUMID_NH'   :0xF5,
    'WRITE'             :0xE6,
    'READ'              :0xE7,
    'RESET'             :0xFE,
    'SERIAL'            :[0xFA,0x0F,0xFC,0xC9],
    'FIRMWARE_READ'     :[0x84,0xB8]

}

HTU21D_RESOLUTION_MASK = 0x7E;
HTU21D_RESOLUTIONS = {
    'TEMP14_HUM12':int('00000000',2),
    'TEMP13_HUM10':int('10000000',2),
    'TEMP12_HUM08':int('00000001',2),
    'TEMP11_HUM11':int('10000001',2)
}

# Device with conversion time for tmp/resolution bit
# The format is:
#  <CHIPNAME>:{id:<ID>, ..<RESOlUTION>:[<temp time>,<humidity time>].. }
HTU21D_DEVICES = {
    'SI7013':{'id':0x0D,
        'TEMP14_HUM12':[.11,.12],
        'TEMP13_HUM10':[ .7, .5],
        'TEMP12_HUM08':[ .4, .4],
        'TEMP11_HUM11':[ .3, .7]},
    'SI7020':{'id':0x14,
        'TEMP14_HUM12':[.11,.12],
        'TEMP13_HUM10':[ .7, .5],
        'TEMP12_HUM08':[ .4, .4],
        'TEMP11_HUM11':[ .3, .7]},
    'SI7021':{'id':0x14,
        'TEMP14_HUM12':[.11,.12],
        'TEMP13_HUM10':[ .7, .5],
        'TEMP12_HUM08':[ .4, .4],
        'TEMP11_HUM11':[ .3, .7]},
    'SHT21': {'id':0x31,
        'TEMP14_HUM12':[.85,.29],
        'TEMP13_HUM10':[.43, .9],
        'TEMP12_HUM08':[.22, .4],
        'TEMP11_HUM11':[.11,.15]},
    'HTU21D':{'id':0x32,
        'TEMP14_HUM12':[.50,.16],
        'TEMP13_HUM10':[.25, .5],
        'TEMP12_HUM08':[.13, .3],
        'TEMP11_HUM11':[.12, .8]}
}
#temperature coefficient for RH compensation at range 0C..80C,
#  for HTU21D & SHT21 only
HTU21D_TEMP_COEFFICIENT= -0.15
#crc8 polynomial for 16bit value, CRC8 -> x^8 + x^5 + x^4 + 1
HTU21D_CRC8_POLYNOMINAL= 0x13100

class HTU21D:
    def __init__(self, config):
        self.printer = config.get_printer()
        self.name = config.get_name().split()[-1]
        self.reactor = self.printer.get_reactor()
        self.i2c = bus.MCU_I2C_from_config(
            config, default_addr=HTU21D_I2C_ADDR, default_speed=100000)
        self.hold_master_mode = config.getboolean('htu21d_hold_master',False)
        self.resolution = config.get('htu21d_resolution','TEMP12_HUM08')
        self.report_time = config.getint('htu21d_report_time',30,minval=5)
        if self.resolution not in HTU21D_RESOLUTIONS:
            raise config.error("Invalid HTU21D Resolution. Valid are %s"
                % '|'.join(HTU21D_RESOLUTIONS.keys()))
        self.deviceId = config.get('sensor_type')
        self.temp = self.min_temp = self.max_temp = self.humidity = 0.
        self.sample_timer = self.reactor.register_timer(self._sample_htu21d)
        self.printer.add_object("htu21d " + self.name, self)
        self.printer.register_event_handler("klippy:connect",
                                            self.handle_connect)

    def handle_connect(self):
        self._init_htu21d()
        self.reactor.update_timer(self.sample_timer, self.reactor.NOW)

    def setup_minmax(self, min_temp, max_temp):
        self.min_temp = min_temp
        self.max_temp = max_temp

    def setup_callback(self, cb):
        self._callback = cb

    def get_report_time_delta(self):
        return self.report_time

    def _init_htu21d(self):
        # Device Soft Reset
        self.i2c.i2c_write([HTU21D_COMMANDS['RESET']])
        # Wait 15ms after reset
        self.reactor.pause(self.reactor.monotonic() + .15)

        # Read ChipId
        params = self.i2c.i2c_read([HTU21D_COMMANDS['SERIAL'][2],
                                    HTU21D_COMMANDS['SERIAL'][3]], 3)
        response = bytearray(params['response'])
        rdevId = response[0] << 8
        rdevId |= response[1]
        checksum = response[2]
        if self._chekCRC8(rdevId) != checksum:
            logging.warn("htu21d: Reading deviceId !Checksum error!")
        rdevId = rdevId >> 8
        deviceId_list = list(
            filter(
              lambda elem: HTU21D_DEVICES[elem]['id'] == rdevId,HTU21D_DEVICES)
            )
        if len(deviceId_list) != 0:
            logging.info("htu21d: Found Device Type %s" % deviceId_list[0])
        else:
            logging.warn("htu21d: Unknown Device ID %#x " % rdevId)

        if(self.deviceId != deviceId_list[0]):
            logging.warn(
                "htu21d: Found device %s. Forcing to type %s as config.",
                 deviceId_list[0],self.deviceId)

        # Set Resolution
        params = self.i2c.i2c_read([HTU21D_COMMANDS['READ']], 1)
        response = bytearray(params['response'])
        registerData = response[0] & HTU21D_RESOLUTION_MASK
        registerData |= HTU21D_RESOLUTIONS[self.resolution]
        self.i2c.i2c_write([HTU21D_COMMANDS['WRITE']],registerData)
        logging.info("htu21d: Setting resolution to %s " % self.resolution)

    def _sample_htu21d(self, eventtime):
        try:
            # Read Temeprature
            if self.hold_master_mode:
                params = self.i2c.i2c_write([HTU21D_COMMANDS['HTU21D_TEMP']])
            else:
                params = self.i2c.i2c_write([HTU21D_COMMANDS['HTU21D_TEMP_NH']])

            # Wait
            self.reactor.pause(self.reactor.monotonic()
            + HTU21D_DEVICES[self.deviceId][self.resolution][0])


            params = self.i2c.i2c_read([],3)

            response = bytearray(params['response'])
            rtemp  = response[0] << 8
            rtemp |= response[1]
            if self._chekCRC8(rtemp) != response[2]:
                logging.warn("htu21d: Checksum error on Temperature reading!")
            else:
                self.temp = (0.002681 * float(rtemp) - 46.85)
                logging.debug("htu21d: Temperature %.2f " % self.temp)

            # Read Humidity
            if self.hold_master_mode:
                self.i2c.i2c_write([HTU21D_COMMANDS['HTU21D_HUMID']])
            else:
                self.i2c.i2c_write([HTU21D_COMMANDS['HTU21D_HUMID_NH']])

            # Wait
            self.reactor.pause(self.reactor.monotonic()
            + HTU21D_DEVICES[self.deviceId][self.resolution][1])

            params = self.i2c.i2c_read([],3)

            response = bytearray(params['response'])
            rhumid = response[0] << 8
            rhumid|= response[1]
            if self._chekCRC8(rhumid) != response[2]:
                logging.warn("htu21d: Checksum error on Humidity reading!")
            else:
                #clear status bits,
                # humidity always returns xxxxxx10 in the LSB field
                rhumid   ^= 0x02;
                self.humidity = (0.001907 * float(rhumid) - 6)
                if (self.humidity < 0):
                    #due to RH accuracy, measured value might be
                    # slightly less than 0 or more 100
                    self.humidity = 0
                elif (self.humidity > 100):
                    self.humidity = 100
                # Only for HTU21D & SHT21.
                # Calculates temperature compensated Humidity, %RH
                if( self.deviceId in ['SHT21','HTU21D']
                    and self.temp > 0 and self.temp < 80):
                    logging.debug("htu21d: Do temp compensation..")
                    self.humidity = self.humidity
                    + (25.0 - self.temp) * HTU21D_TEMP_COEFFICIENT;
                logging.debug("htu21d: Humidity %.2f " % self.humidity)
        except Exception:
            logging.exception("htu21d: Error reading data")
            self.temp = self.humidity = .0
            return self.reactor.NEVER

        if self.temp < self.min_temp or self.temp > self.max_temp:
            self.printer.invoke_shutdown(
                "HTU21D temperature %0.1f outside range of %0.1f:%.01f"
                % (self.temp, self.min_temp, self.max_temp))

        measured_time = self.reactor.monotonic()
        print_time = self.i2c.get_mcu().estimated_print_time(measured_time)
        self._callback(print_time, self.temp)
        return measured_time + self.report_time

    def _chekCRC8(self,data):
        for bit in range(0,16):
            if (data & 0x8000):
                data = (data << 1) ^ HTU21D_CRC8_POLYNOMINAL;
            else:
                data <<= 1
        data = data >> 8
        return data

    def get_status(self, eventtime):
        return {
            'temperature': self.temp,
            'humidity': self.humidity,
        }


def load_config(config):
    # Register sensor
    pheater = config.get_printer().lookup_object("heaters")
    for stype in HTU21D_DEVICES:
        pheater.add_sensor_factory(stype, HTU21D)