Implementation of 1-Wire with added Alarm Search Functionality

Dependents:   Max32630_One_Wire_Interface

OneWire_Masters/DS248x/ds248x.cpp

Committer:
IanBenzMaxim
Date:
2016-05-09
Revision:
72:6892702709ee
Parent:
71:562f5c702094

File content as of revision 72:6892702709ee:

/******************************************************************//**
* Copyright (C) 2016 Maxim Integrated Products, Inc., All Rights Reserved.
*
* 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 MAXIM INTEGRATED 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.
*
* Except as contained in this notice, the name of Maxim Integrated
* Products, Inc. shall not be used except as stated in the Maxim Integrated
* Products, Inc. Branding Policy.
*
* The mere transfer of this software does not imply any licenses
* of trade secrets, proprietary technology, copyrights, patents,
* trademarks, maskwork rights, or any other form of intellectual
* property whatsoever. Maxim Integrated Products, Inc. retains all
* ownership rights.
**********************************************************************/


#include "ds248x.h"
#include "RomId.hpp"


enum DS248X_STATUS_BITS
{
    STATUS_1WB  = 0x01,
    STATUS_PPD = 0x02,
    STATUS_SD = 0x04,
    STATUS_LL = 0x08,
    STATUS_RST = 0x10,
    STATUS_SBR = 0x20,
    STATUS_TSB = 0x40,
    STATUS_DIR = 0x80
};

static const int I2C_WRITE_OK = 0;
static const int I2C_READ_OK = 0;


std::uint8_t Ds248x::Config::readByte() const
{
  std::uint8_t config = 0;
  if (get1WS())
    config |= 0x08;
  if (getSPU())
    config |= 0x04;
  if (getPDN())
    config |= 0x02;
  if (getAPU())
    config |= 0x01;
  return config;
}

std::uint8_t Ds248x::Config::writeByte() const
{
  std::uint8_t config = readByte();
  return ((~config << 4) | config);
}

void Ds248x::Config::reset()
{
  set1WS(false);
  setSPU(false);
  setPDN(false);
  setAPU(true);
}


//*********************************************************************
Ds248x::Ds248x(I2C &i2c_bus, DS248X_I2C_ADRS adrs)
:_p_i2c_bus(&i2c_bus), _i2c_owner(false)
{
    set_i2c_adrs(adrs);
}


//*********************************************************************
Ds248x::Ds248x(PinName sda, PinName scl, DS248X_I2C_ADRS adrs)
:_p_i2c_bus(new I2C(sda, scl)), _i2c_owner(true)
{
    set_i2c_adrs(adrs);
}


//*********************************************************************
Ds248x::~Ds248x()
{
    if(_i2c_owner)
    {
        delete _p_i2c_bus;
    }
}


//*********************************************************************
OneWireMaster::CmdResult Ds248x::OWInitMaster(void)
{
    OneWireMaster::CmdResult result;

    // reset DS2465 
    result = reset();
    if (result != OneWireMaster::Success)
    return result;

    // write the default configuration setup
    Config defaultConfig;
    result = writeConfig(defaultConfig, true);
    return result;
}


//*********************************************************************
OneWireMaster::CmdResult Ds248x::reset(void)
{    
    // Device Reset
    //   S AD,0 [A] DRST [A] Sr AD,1 [A] [SS] A\ P
    //  [] indicates from slave
    //  SS status byte to read to verify state
    
  OneWireMaster::CmdResult result;
  std::uint8_t buf;
  
  result = sendCommand(CMD_DRST);
  
  if (result == OneWireMaster::Success)
    result = readRegister(REG_STATUS, buf, true);

  if (result == OneWireMaster::Success)
  {
    if ((buf & 0xF7) != 0x10)
      result = OneWireMaster::OperationFailure;
  }
   
  if (result == OneWireMaster::Success)
    OWReset(); // do a command to get 1-Wire master reset out of holding state

  return result; 
}


//*********************************************************************
OneWireMaster::CmdResult Ds248x::channelSelect(uint8_t channel)
{
    OneWireMaster::CmdResult result;
    uint8_t ch, ch_read;
    
    // Channel Select (Case A)
    //   S AD,0 [A] CHSL [A] CC [A] Sr AD,1 [A] [RR] A\ P
    //  [] indicates from slave
    //  CC channel value
    //  RR channel read back
    
    switch (channel)
    {
      default: case 0: ch = 0xF0; ch_read = 0xB8; break;
      case 1: ch = 0xE1; ch_read = 0xB1; break;
      case 2: ch = 0xD2; ch_read = 0xAA; break;
      case 3: ch = 0xC3; ch_read = 0xA3; break;
      case 4: ch = 0xB4; ch_read = 0x9C; break;
      case 5: ch = 0xA5; ch_read = 0x95; break;
      case 6: ch = 0x96; ch_read = 0x8E; break;
      case 7: ch = 0x87; ch_read = 0x87; break;
    };
    
    result = sendCommand(CMD_CHSL, ch);
    if (result == OneWireMaster::Success)
    {
        result = readRegister(REG_CHANNEL_SELECT, ch, true);
        if (result == OneWireMaster::Success)
        {
            // check for failure due to incorrect read back of channel
            if (ch != ch_read)
            {
                result = OneWireMaster::OperationFailure;
            }
        }
    }
    
    return result;
}


//*********************************************************************
OneWireMaster::CmdResult Ds248x::adjustTiming(DS2484_ADJST_PARAMS param, uint8_t val)
{
    OneWireMaster::CmdResult result;
    
    uint8_t read_port_config;
    uint8_t control_byte;

    control_byte = (((param & 0x0F) << 4) | (val & 0x0F));
    
    result = sendCommand(CMD_A1WP, control_byte);
    if (result != Success)
        return result;
        
    result = readRegister(REG_PORT_CONFIG, read_port_config, true);
    if (result != Success)
        return result;
        
    if ((control_byte & 0x0F) != read_port_config) 
        result = OneWireMaster::OperationFailure;
    
    return result;
}


//*********************************************************************
OneWireMaster::CmdResult Ds248x::OWTriplet(SearchDirection & search_direction, std::uint8_t & sbr, std::uint8_t & tsb)
{
    // 1-Wire Triplet (Case B)
    //   S AD,0 [A] 1WT [A] SS [A] Sr AD,1 [A] [Status] A [Status] A\ P
    //                                         \--------/
    //                           Repeat until 1WB bit has changed to 0
    //  [] indicates from slave
    //  SS indicates byte containing search direction bit value in msbit

  OneWireMaster::CmdResult result;
  result = sendCommand(CMD_1WT, (std::uint8_t)((search_direction == DIRECTION_WRITE_ONE) ? 0x80 : 0x00));
  if (result == OneWireMaster::Success)
  {
    std::uint8_t status;
    result = pollBusy(&status);
    if (result == OneWireMaster::Success)
    {
      // check bit results in status byte
      sbr = ((status & STATUS_SBR) == STATUS_SBR);
      tsb = ((status & STATUS_TSB) == STATUS_TSB);
      search_direction = ((status & STATUS_DIR) == STATUS_DIR) ? DIRECTION_WRITE_ONE : DIRECTION_WRITE_ZERO;
    }
  }
  return result;
}


//*********************************************************************
OneWireMaster::CmdResult Ds248x::OWReset(void)
{
    // 1-Wire reset (Case B)
    //   S AD,0 [A] 1WRS [A] Sr AD,1 [A] [Status] A [Status] A\ P
    //                                   \--------/
    //                       Repeat until 1WB bit has changed to 0
    //  [] indicates from slave

  OneWireMaster::CmdResult result;
  std::uint8_t buf;
    
  result = sendCommand(CMD_1WRS);

  if (result == OneWireMaster::Success)
    result = pollBusy(&buf);
  
  if (result == OneWireMaster::Success)
  {       
    // check for presence detect
    if ((buf & STATUS_PPD) != STATUS_PPD)
      result = OneWireMaster::OperationFailure;
  }

  return result;
}


//*********************************************************************
OneWireMaster::CmdResult Ds248x::OWTouchBitSetLevel(uint8_t & sendrecvbit, OWLevel after_level)
{
    // 1-Wire bit (Case B)
    //   S AD,0 [A] 1WSB [A] BB [A] Sr AD,1 [A] [Status] A [Status] A\ P
    //                                          \--------/
    //                           Repeat until 1WB bit has changed to 0
    //  [] indicates from slave
    //  BB indicates byte containing bit value in msbit

  OneWireMaster::CmdResult result;
  
  result = configureLevel(after_level);
  if (result != OneWireMaster::Success)
    return result;
  
  std::uint8_t status;
  
  result = sendCommand(CMD_1WSB, (std::uint8_t)(sendrecvbit ? 0x80 : 0x00));
  
  if (result == OneWireMaster::Success)
    result = pollBusy(&status);
 
  if (result == OneWireMaster::Success)
    sendrecvbit = (status & STATUS_SBR);
  
  return result;
}


//*********************************************************************
OneWireMaster::CmdResult Ds248x::OWWriteByteSetLevel(uint8_t sendbyte, OWLevel after_level)
{
    // 1-Wire Write Byte (Case B)
    //   S AD,0 [A] 1WWB [A] DD [A] Sr AD,1 [A] [Status] A [Status] A\ P
    //                                          \--------/
    //                             Repeat until 1WB bit has changed to 0
    //  [] indicates from slave
    //  DD data to write

  OneWireMaster::CmdResult result;
  
  result = configureLevel(after_level);
  if (result != OneWireMaster::Success)
    return result;
  
  result = sendCommand(CMD_1WWB, sendbyte);
  if (result == OneWireMaster::Success)
    result = pollBusy();

  return result;
}


//*********************************************************************
OneWireMaster::CmdResult Ds248x::OWReadByteSetLevel(uint8_t & recvbyte, OWLevel after_level)
{
    // 1-Wire Read Bytes (Case C)
    //   S AD,0 [A] 1WRB [A] Sr AD,1 [A] [Status] A [Status] A\
    //                                   \--------/
    //                     Repeat until 1WB bit has changed to 0
    //   Sr AD,0 [A] SRP [A] E1 [A] Sr AD,1 [A] DD A\ P
    //
    //  [] indicates from slave
    //  DD data read

  OneWireMaster::CmdResult result;
  std::uint8_t buf;
  
  result = configureLevel(after_level);
  if (result != OneWireMaster::Success)
    return result;
   
  result = sendCommand(CMD_1WRB);

  if (result == OneWireMaster::Success)
    result = pollBusy();

  if (result == OneWireMaster::Success)
    result = readRegister(REG_READ_DATA, buf);
  
  if (result == OneWireMaster::Success)
    recvbyte = buf;
   
  return result;
}


//*********************************************************************
OneWireMaster::CmdResult Ds248x::OWSetSpeed(OWSpeed new_speed)
{
  // Requested speed is already set
  if (m_curConfig.get1WS() == (new_speed == SPEED_OVERDRIVE))
    return OneWireMaster::Success;
       
  // set the speed
  Config newConfig = m_curConfig;
  newConfig.set1WS(new_speed == SPEED_OVERDRIVE);

  // write the new config
  return writeConfig(newConfig, true);
}


//*********************************************************************
OneWireMaster::CmdResult Ds248x::OWSetLevel(OWLevel new_level)
{
    if (new_level == LEVEL_STRONG)
        return OneWireMaster::OperationFailure;
  
    return configureLevel(new_level);
}


//*********************************************************************
void Ds248x::set_i2c_adrs(DS248X_I2C_ADRS adrs)
{
    _w_adrs = (adrs << 1);
    _r_adrs = (_w_adrs | 1);
}




OneWireMaster::CmdResult Ds248x::writeConfig(const Config & config, bool verify)
{
  std::uint8_t configBuf;
  OneWireMaster::CmdResult result;
   
  configBuf = config.writeByte();
  result = sendCommand(CMD_WCFG, configBuf);
  if (verify)
  {
    if (result == OneWireMaster::Success)
    {
      result = readRegister(REG_CONFIG, configBuf);
    }
    if (result == OneWireMaster::Success)
    {
      if (configBuf != config.readByte())
        result = OneWireMaster::OperationFailure;
    }
  }
  
  if (result == OneWireMaster::Success)
    m_curConfig = config;

  return result;
}

OneWireMaster::CmdResult Ds248x::readRegister(DS248X_REG reg, std::uint8_t & buf, bool skipSetPointer) const
{
    CmdResult result;
    result = sendCommand(CMD_SRP, reg);
    if (result == Success)
    {
        if (_p_i2c_bus->read(_r_adrs, reinterpret_cast<char *>(&buf), 1) != I2C_READ_OK)
            result = CommunicationReadError;
    }
    return result;
}

OneWireMaster::CmdResult Ds248x::pollBusy(std::uint8_t * pStatus)
{
  const unsigned int pollLimit = 200;

  OneWireMaster::CmdResult result;
  std::uint8_t status;
  unsigned int pollCount = 0;
   
  do
  {
    result = readRegister(REG_STATUS, status, true);
    if (result != OneWireMaster::Success)
      return result;
    if (pStatus != NULL)
        *pStatus = status;
    if (pollCount++ >= pollLimit)
      return OneWireMaster::TimeoutError;
  } while (status & STATUS_1WB);

  return OneWireMaster::Success;
}

OneWireMaster::CmdResult Ds248x::configureLevel(OWLevel level)
{
    OneWireMaster::CmdResult result;
    if (m_curConfig.getSPU() != (level == LEVEL_STRONG))
    {
        Config newConfig = m_curConfig;
        newConfig.setSPU(level == LEVEL_STRONG);
        result = writeConfig(newConfig, true);
    }
    else
    {
        result = OneWireMaster::Success;
    }
    return result;
}

OneWireMaster::CmdResult Ds248x::sendCommand(DS248X_CMDS cmd) const
{
    CmdResult result;
    if (_p_i2c_bus->write(_w_adrs, reinterpret_cast<const char *>(&cmd), 1) == I2C_WRITE_OK)
        result = Success;
    else
        result = CommunicationWriteError;
    return result;
}

OneWireMaster::CmdResult Ds248x::sendCommand(DS248X_CMDS cmd, std::uint8_t param) const
{
    CmdResult result;
    char buf[2] = { cmd, param };
    if (_p_i2c_bus->write(_w_adrs, buf, 2) == I2C_WRITE_OK)
        result = Success;
    else
        result = CommunicationWriteError;
    return result;
}