summaryrefslogblamecommitdiffstats
path: root/cpukit/dev/i2c/gpio-nxp-pca9535.c
blob: 32ef8847d759ebb2b2adf1777e0aa9d582bc23c5 (plain) (tree)
1
2
3
4
5
6
7
8
9








                                                
                                                   
  




                                                          

                    




















                                     

                            



                           

                                    


                           

                                   

     
          
 



                                                                     







                                    
                                                                 



                           

                                    








































































                                                                                 
/**
 * @file
 *
 * @brief GPIO NXP PCA9535 Driver Implementation
 *
 * @ingroup I2CGPIONXPPCA9535
 */

/*
 * Copyright (c) 2014 embedded brains GmbH & Co. KG
 *
 * The license and distribution terms for this file may be
 * found in the file LICENSE in this distribution or at
 * http://www.rtems.org/license/LICENSE.
 */

#ifdef HAVE_CONFIG_H
#include "config.h"
#endif

#include <dev/i2c/gpio-nxp-pca9535.h>

typedef enum {
  GPIO_NXP_PCA9535_INPUT_PORT_0,
  GPIO_NXP_PCA9535_INPUT_PORT_1,
  GPIO_NXP_PCA9535_OUTPUT_PORT_0,
  GPIO_NXP_PCA9535_OUTPUT_PORT_1,
  GPIO_NXP_PCA9535_POL_INV_PORT_0,
  GPIO_NXP_PCA9535_POL_INV_PORT_1,
  GPIO_NXP_PCA9535_CONF_PORT_0,
  GPIO_NXP_PCA9535_CONF_PORT_1
} gpio_nxp_pca9535_port;

static int gpio_nxp_pca9535_get_reg(
  i2c_dev *dev,
  gpio_nxp_pca9535_port port,
  uint16_t *val
)
{
  uint8_t out[1] = { port };
  uint8_t in[sizeof(*val)];
  i2c_msg msgs[2] = {
    {
      .addr = dev->address,
      .flags = 0,
      .len = (uint16_t) sizeof(out),
      .buf = &out[0]
    }, {
      .addr = dev->address,
      .flags = I2C_M_RD,
      .len = (uint16_t) sizeof(in),
      .buf = &in[0]
    }
  };
  int err;

  err = i2c_bus_transfer(dev->bus, &msgs[0], RTEMS_ARRAY_SIZE(msgs));
  *val = in[0] | (in[1] << 8);

  return err;
}

static int gpio_nxp_pca9535_set_reg(
  i2c_dev *dev,
  gpio_nxp_pca9535_port port,
  uint16_t val
)
{
  uint8_t out[3] = { port, (uint8_t) val, (uint8_t) (val >> 8) };
  i2c_msg msgs[1] = {
    {
      .addr = dev->address,
      .flags = 0,
      .len = (uint16_t) sizeof(out),
      .buf = &out[0]
    }
  };

  return i2c_bus_transfer(dev->bus, &msgs[0], RTEMS_ARRAY_SIZE(msgs));
}

static int gpio_nxp_pca9535_ioctl(
  i2c_dev *dev,
  ioctl_command_t command,
  void *arg
)
{
  uint16_t v16 = (uint16_t)(uintptr_t) arg;
  uint32_t v32 = (uint32_t)(uintptr_t) arg;
  int err;

  switch (command) {
    case GPIO_NXP_PCA9535_GET_INPUT:
      err = gpio_nxp_pca9535_get_reg(dev, GPIO_NXP_PCA9535_INPUT_PORT_0, arg);
      break;
    case GPIO_NXP_PCA9535_GET_OUTPUT:
      err = gpio_nxp_pca9535_get_reg(dev, GPIO_NXP_PCA9535_OUTPUT_PORT_0, arg);
      break;
    case GPIO_NXP_PCA9535_CLEAR_AND_SET_OUTPUT:
      i2c_bus_obtain(dev->bus);
      err = gpio_nxp_pca9535_get_reg(dev, GPIO_NXP_PCA9535_OUTPUT_PORT_0, &v16);
      if (err == 0) {
        v16 &= ~((uint16_t) v32);
        v16 |= (uint16_t) (v32 >> 16);
        err = gpio_nxp_pca9535_set_reg(dev, GPIO_NXP_PCA9535_OUTPUT_PORT_0, v16);
      }
      i2c_bus_release(dev->bus);
      break;
    case GPIO_NXP_PCA9535_SET_OUTPUT:
      err = gpio_nxp_pca9535_set_reg(dev, GPIO_NXP_PCA9535_OUTPUT_PORT_0, v16);
      break;
    case GPIO_NXP_PCA9535_GET_POL_INV:
      err = gpio_nxp_pca9535_get_reg(dev, GPIO_NXP_PCA9535_POL_INV_PORT_0, arg);
      break;
    case GPIO_NXP_PCA9535_SET_POL_INV:
      err = gpio_nxp_pca9535_set_reg(dev, GPIO_NXP_PCA9535_POL_INV_PORT_0, v16);
      break;
    case GPIO_NXP_PCA9535_GET_CONFIG:
      err = gpio_nxp_pca9535_get_reg(dev, GPIO_NXP_PCA9535_CONF_PORT_0, arg);
      break;
    case GPIO_NXP_PCA9535_SET_CONFIG:
      err = gpio_nxp_pca9535_set_reg(dev, GPIO_NXP_PCA9535_CONF_PORT_0, v16);
      break;
    default:
      err = -ENOTTY;
      break;
  }

  return err;
}

int i2c_dev_register_gpio_nxp_pca9535(
  const char *bus_path,
  const char *dev_path,
  uint16_t address
)
{
  i2c_dev *dev;

  dev = i2c_dev_alloc_and_init(sizeof(*dev), bus_path, address);
  if (dev == NULL) {
    return -1;
  }

  dev->ioctl = gpio_nxp_pca9535_ioctl;

  return i2c_dev_register(dev, dev_path);
}