| /**************************************************************************** |
| * drivers/motor/foc/drv8301.c |
| * |
| * Licensed to the Apache Software Foundation (ASF) under one or more |
| * contributor license agreements. See the NOTICE file distributed with |
| * this work for additional information regarding copyright ownership. The |
| * ASF licenses this file to you under the Apache License, Version 2.0 (the |
| * "License"); you may not use this file except in compliance with the |
| * License. You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT |
| * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the |
| * License for the specific language governing permissions and limitations |
| * under the License. |
| * |
| ****************************************************************************/ |
| |
| /**************************************************************************** |
| * Included Files |
| ****************************************************************************/ |
| |
| #include <nuttx/config.h> |
| |
| #include <debug.h> |
| |
| #include <nuttx/kmalloc.h> |
| #include <nuttx/nuttx.h> |
| |
| #include <nuttx/spi/spi.h> |
| |
| #include <nuttx/motor/foc/drv8301.h> |
| #include <nuttx/motor/motor_ioctl.h> |
| |
| /**************************************************************************** |
| * Pre-processor Definitions |
| ****************************************************************************/ |
| |
| #if FOC_BOARDCFG_GAINLIST_LEN < 4 |
| # error FOC_BOARDCFG_GAINLIST_LEN < 4 not supported |
| #endif |
| |
| /**************************************************************************** |
| * Private Types |
| ****************************************************************************/ |
| |
| /* DRV8301 device */ |
| |
| struct drv8301_priv_s |
| { |
| /* Common FOC power-stage driver - must be first */ |
| |
| struct focpwr_dev_s dev; |
| |
| FAR struct drv8301_ops_s *ops; /* Board ops */ |
| |
| FAR struct spi_dev_s *spi; /* SPI device reference */ |
| FAR struct drv8301_cfg_s cfg; /* Configuration */ |
| }; |
| |
| /**************************************************************************** |
| * Private Function Prototypes |
| ****************************************************************************/ |
| |
| static int drv8301_fault_isr(int irq, void *context, void *arg); |
| |
| static int drv8301_gain_set(FAR struct focpwr_dev_s *dev, int gain); |
| static int drv8301_gain_get(FAR struct focpwr_dev_s *dev, FAR int *gain); |
| |
| static int drv8301_setup(FAR struct focpwr_dev_s *dev); |
| static int drv8301_shutdown(FAR struct focpwr_dev_s *dev); |
| static int drv8301_calibration(FAR struct focpwr_dev_s *dev, bool state); |
| static int drv8301_ioctl(FAR struct focpwr_dev_s *dev, int cmd, |
| unsigned long arg); |
| |
| /**************************************************************************** |
| * Private Data |
| ****************************************************************************/ |
| |
| struct focpwr_ops_s g_drv8301_ops = |
| { |
| .setup = drv8301_setup, |
| .shutdown = drv8301_shutdown, |
| .calibration = drv8301_calibration, |
| .ioctl = drv8301_ioctl, |
| }; |
| |
| /**************************************************************************** |
| * Private Functions |
| ****************************************************************************/ |
| |
| /**************************************************************************** |
| * Name: drv8301_lock |
| ****************************************************************************/ |
| |
| static void drv8301_lock(FAR struct drv8301_priv_s *priv) |
| { |
| SPI_LOCK(priv->spi, 1); |
| SPI_SETBITS(priv->spi, 16); |
| SPI_SETMODE(priv->spi, SPIDEV_MODE1); |
| SPI_SETFREQUENCY(priv->spi, priv->cfg.freq); |
| } |
| |
| /**************************************************************************** |
| * Name: drv8301_unlock |
| ****************************************************************************/ |
| |
| static void drv8301_unlock(FAR struct drv8301_priv_s *priv) |
| { |
| SPI_LOCK(priv->spi, 0); |
| } |
| |
| /**************************************************************************** |
| * Name: drv8301_read |
| ****************************************************************************/ |
| |
| static void drv8301_read(FAR struct drv8301_priv_s *priv, uint8_t addr, |
| uint16_t *data) |
| { |
| uint16_t regval = 0; |
| |
| drv8301_lock(priv); |
| SPI_SELECT(priv->spi, SPIDEV_MOTOR(priv->dev.devno), true); |
| |
| /* Read command */ |
| |
| regval |= (1 << 15); |
| regval |= ((addr & 0x0f) << 11); |
| |
| /* Send command */ |
| |
| SPI_SEND(priv->spi, regval); |
| |
| /* Toggle CS pin, otherwise read doesn't work */ |
| |
| SPI_SELECT(priv->spi, SPIDEV_MOTOR(priv->dev.devno), false); |
| SPI_SELECT(priv->spi, SPIDEV_MOTOR(priv->dev.devno), true); |
| |
| /* Read output */ |
| |
| regval = 0; |
| SPI_RECVBLOCK(priv->spi, ®val, 1); |
| |
| /* Retrun data */ |
| |
| *data = (regval & 0x7ff); |
| |
| SPI_SELECT(priv->spi, SPIDEV_MOTOR(priv->dev.devno), false); |
| drv8301_unlock(priv); |
| } |
| |
| /**************************************************************************** |
| * Name: drv8301_write |
| ****************************************************************************/ |
| |
| static void drv8301_write(FAR struct drv8301_priv_s *priv, uint8_t addr, |
| uint16_t data) |
| { |
| uint16_t regval = 0; |
| |
| drv8301_lock(priv); |
| SPI_SELECT(priv->spi, SPIDEV_MOTOR(priv->dev.devno), true); |
| |
| /* Write command */ |
| |
| regval |= (0 << 15); |
| regval |= ((addr & 0x0f) << 11); |
| regval |= (0x7ff & data); |
| |
| /* Send data */ |
| |
| SPI_SEND(priv->spi, regval); |
| |
| SPI_SELECT(priv->spi, SPIDEV_MOTOR(priv->dev.devno), false); |
| drv8301_unlock(priv); |
| } |
| |
| /**************************************************************************** |
| * Name: drv8301_fault_isr |
| ****************************************************************************/ |
| |
| static int drv8301_fault_isr(int irq, FAR void *context, void *arg) |
| { |
| FAR struct drv8301_priv_s *priv = NULL; |
| |
| priv = (struct drv8301_priv_s *)arg; |
| DEBUGASSERT(priv != NULL); |
| |
| priv->ops->fault_handle(&priv->dev); |
| |
| return OK; |
| } |
| |
| /**************************************************************************** |
| * Name: drv8301_setup |
| ****************************************************************************/ |
| |
| static int drv8301_setup(FAR struct focpwr_dev_s *dev) |
| { |
| FAR struct drv8301_priv_s *priv = (FAR struct drv8301_priv_s *)dev; |
| uint16_t status1 = 0; |
| uint16_t status2 = 0; |
| uint16_t ctrl1 = 0; |
| uint16_t ctrl2 = 0; |
| int ret = OK; |
| |
| /* Reset chip */ |
| |
| priv->ops->gate_enable(dev, true); |
| up_udelay(30); |
| priv->ops->gate_enable(dev, false); |
| up_udelay(30); |
| priv->ops->gate_enable(dev, true); |
| up_mdelay(10); |
| |
| /* Attach fault handler */ |
| |
| priv->ops->fault_attach(dev, drv8301_fault_isr, priv); |
| |
| /* Get status registers */ |
| |
| drv8301_read(priv, DRV8301_REG_STAT1, &status1); |
| drv8301_read(priv, DRV8301_REG_STAT2, &status2); |
| |
| /* Configure CTRL1 */ |
| |
| ctrl1 = DRV8301_CTRL1_GCURR(priv->cfg.gate_curr); |
| ctrl1 |= DRV8301_CTRL1_OCADJ(priv->cfg.oc_adj); |
| ctrl1 |= (priv->cfg.pwm_mode ? DRV8301_CTRL1_PWMMODE : 0); |
| drv8301_write(priv, DRV8301_REG_CTRL1, ctrl1); |
| |
| /* Configure CTRL2 */ |
| |
| ctrl2 = DRV8301_CTRL2_GAIN(priv->cfg.gain); |
| drv8301_write(priv, DRV8301_REG_CTRL2, ctrl2); |
| |
| return ret; |
| } |
| |
| /**************************************************************************** |
| * Name: drv8301_shutdown |
| ****************************************************************************/ |
| |
| static int drv8301_shutdown(FAR struct focpwr_dev_s *dev) |
| { |
| FAR struct drv8301_priv_s *priv = (FAR struct drv8301_priv_s *)dev; |
| |
| /* Disable chip */ |
| |
| priv->ops->gate_enable(dev, false); |
| |
| /* Disable nFAULT interrupt */ |
| |
| priv->ops->fault_attach(dev, NULL, NULL); |
| |
| return OK; |
| } |
| |
| /**************************************************************************** |
| * Name: drv8301_gain_get |
| ****************************************************************************/ |
| |
| static int drv8301_gain_get(FAR struct focpwr_dev_s *dev, FAR int *gain) |
| { |
| FAR struct drv8301_priv_s *priv = (FAR struct drv8301_priv_s *)dev; |
| uint16_t ctrl2 = 0; |
| int ret = OK; |
| |
| drv8301_read(priv, DRV8301_REG_CTRL2, &ctrl2); |
| ctrl2 &= DRV8301_CTRL2_GAIN_MASK; |
| |
| if (ctrl2 == DRV8301_CTRL2_GAIN_10) |
| { |
| *gain = 10; |
| } |
| else if (ctrl2 == DRV8301_CTRL2_GAIN_20) |
| { |
| *gain = 20; |
| } |
| else if (ctrl2 == DRV8301_CTRL2_GAIN_40) |
| { |
| *gain = 40; |
| } |
| else if (ctrl2 == DRV8301_CTRL2_GAIN_80) |
| { |
| *gain = 80; |
| } |
| else |
| { |
| ret = -EINVAL; |
| } |
| |
| return ret; |
| } |
| |
| /**************************************************************************** |
| * Name: drv8301_gain_set |
| ****************************************************************************/ |
| |
| static int drv8301_gain_set(FAR struct focpwr_dev_s *dev, int gain) |
| { |
| FAR struct drv8301_priv_s *priv = (FAR struct drv8301_priv_s *)dev; |
| uint16_t ctrl2 = 0; |
| int ret = OK; |
| |
| drv8301_read(priv, DRV8301_REG_CTRL2, &ctrl2); |
| |
| ctrl2 &= ~DRV8301_CTRL2_GAIN_MASK; |
| |
| if (gain == 10) |
| { |
| ctrl2 |= DRV8301_CTRL2_GAIN_10; |
| } |
| else if (gain == 20) |
| { |
| ctrl2 |= DRV8301_CTRL2_GAIN_20; |
| } |
| else if (gain == 40) |
| { |
| ctrl2 |= DRV8301_CTRL2_GAIN_40; |
| } |
| else if (gain == 80) |
| { |
| ctrl2 |= DRV8301_CTRL2_GAIN_80; |
| } |
| else |
| { |
| ret = -EINVAL; |
| goto errout; |
| } |
| |
| /* Write CTRL2 */ |
| |
| drv8301_write(priv, DRV8301_REG_CTRL2, ctrl2); |
| |
| errout: |
| return ret; |
| } |
| |
| /**************************************************************************** |
| * Name: drv8301_calibration |
| ****************************************************************************/ |
| |
| static int drv8301_calibration(FAR struct focpwr_dev_s *dev, bool state) |
| { |
| FAR struct drv8301_priv_s *priv = (FAR struct drv8301_priv_s *)dev; |
| uint16_t regval = 0; |
| |
| drv8301_read(priv, DRV8301_REG_CTRL2, ®val); |
| |
| if (state == true) |
| { |
| regval |= DRV8301_CTRL2_DCCALCH1; |
| regval |= DRV8301_CTRL2_DCCALCH2; |
| } |
| else |
| { |
| regval &= ~DRV8301_CTRL2_DCCALCH1; |
| regval &= ~DRV8301_CTRL2_DCCALCH2; |
| } |
| |
| drv8301_write(priv, DRV8301_REG_CTRL2, regval); |
| |
| return OK; |
| } |
| |
| /**************************************************************************** |
| * Name: drv8301_ioctl |
| ****************************************************************************/ |
| |
| static int drv8301_ioctl(FAR struct focpwr_dev_s *dev, int cmd, |
| unsigned long arg) |
| { |
| int ret = OK; |
| |
| switch (cmd) |
| { |
| case MTRIOC_SET_BOARDCFG: |
| { |
| struct foc_set_boardcfg_s *cfg = |
| (struct foc_set_boardcfg_s *)arg; |
| |
| ret = drv8301_gain_set(dev, cfg->gain); |
| break; |
| } |
| |
| case MTRIOC_GET_BOARDCFG: |
| { |
| struct foc_get_boardcfg_s *cfg = |
| (struct foc_get_boardcfg_s *)arg; |
| |
| ret = drv8301_gain_get(dev, &cfg->gain); |
| |
| cfg->gain_list[0] = 10; |
| cfg->gain_list[1] = 20; |
| cfg->gain_list[2] = 40; |
| cfg->gain_list[3] = 80; |
| |
| break; |
| } |
| |
| default: |
| { |
| ret = -ENOTTY; |
| break; |
| } |
| } |
| |
| return ret; |
| } |
| |
| /**************************************************************************** |
| * Public Functions |
| ****************************************************************************/ |
| |
| /**************************************************************************** |
| * Name: drv8301_register |
| ****************************************************************************/ |
| |
| int drv8301_register(FAR const char *path, |
| FAR struct foc_dev_s *dev, |
| FAR struct drv8301_board_s *board) |
| { |
| FAR struct drv8301_priv_s *priv = NULL; |
| int ret = OK; |
| |
| /* Allocate driver */ |
| |
| priv = kmm_zalloc(sizeof(struct drv8301_priv_s)); |
| if (priv == NULL) |
| { |
| return -ENOMEM; |
| } |
| |
| /* Register FOC device */ |
| |
| ret = foc_register(path, dev); |
| if (ret < 0) |
| { |
| return ret; |
| } |
| |
| /* Store board data */ |
| |
| priv->ops = board->ops; |
| priv->spi = board->spi; |
| |
| /* Store configuration */ |
| |
| memcpy(&priv->cfg, board->cfg, sizeof(struct drv8301_cfg_s)); |
| |
| /* Initialize FOC power stage */ |
| |
| return focpwr_initialize(&priv->dev, board->devno, dev, &g_drv8301_ops); |
| } |