esp32/rmt: Remove outdated RMT driver
The new common RMT driver - already available for ESP32-S2 and ESP32-S3 - will be implemented in another commit for ESP32.
This commit is contained in:
parent
573e1a255f
commit
4b3b22cbeb
@ -296,14 +296,6 @@ config ESP32_PCNT
|
||||
Pulse Counter is currently used to implement Quadracture
|
||||
Encoder.
|
||||
|
||||
config ESP32_RMT
|
||||
bool "Remote Control Module (RMT)"
|
||||
default n
|
||||
depends on EXPERIMENTAL
|
||||
---help---
|
||||
Remote Control Module is currently used to control WS2812
|
||||
RGB LED normally used on LED strips.
|
||||
|
||||
config ESP32_RNG
|
||||
bool "Random Number Generator (RNG)"
|
||||
default n
|
||||
|
@ -85,10 +85,6 @@ ifeq ($(CONFIG_ESP32_PCNT_AS_QE),y)
|
||||
CHIP_CSRCS += esp32_qencoder.c
|
||||
endif
|
||||
|
||||
ifeq ($(CONFIG_ESP32_RMT),y)
|
||||
CHIP_CSRCS += esp32_rmt.c
|
||||
endif
|
||||
|
||||
ifeq ($(CONFIG_ESP32_SPI),y)
|
||||
CHIP_CSRCS += esp32_spi.c
|
||||
ifeq ($(CONFIG_SPI_SLAVE),y)
|
||||
|
@ -1,433 +0,0 @@
|
||||
/****************************************************************************
|
||||
* arch/xtensa/src/esp32/esp32_rmt.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 <stdio.h>
|
||||
#include <sys/types.h>
|
||||
#include <inttypes.h>
|
||||
#include <stdint.h>
|
||||
#include <stdbool.h>
|
||||
#include <assert.h>
|
||||
#include <errno.h>
|
||||
#include <debug.h>
|
||||
|
||||
#include <nuttx/kmalloc.h>
|
||||
#include <arch/board/board.h>
|
||||
#include <nuttx/irq.h>
|
||||
#include <nuttx/arch.h>
|
||||
#include <nuttx/spinlock.h>
|
||||
|
||||
#include "xtensa.h"
|
||||
|
||||
#include "esp32_gpio.h"
|
||||
#include "esp32_rmt.h"
|
||||
#include "esp32_irq.h"
|
||||
#include "esp32_clockconfig.h"
|
||||
|
||||
#include "hardware/esp32_dport.h"
|
||||
#include "hardware/esp32_gpio_sigmap.h"
|
||||
|
||||
#ifdef CONFIG_ESP32_RMT
|
||||
|
||||
/****************************************************************************
|
||||
* Pre-processor Definitions
|
||||
****************************************************************************/
|
||||
|
||||
/****************************************************************************
|
||||
* Private Types
|
||||
****************************************************************************/
|
||||
|
||||
/****************************************************************************
|
||||
* Private Function Prototypes
|
||||
****************************************************************************/
|
||||
|
||||
/* RMT methods */
|
||||
|
||||
static void rmt_reset(struct rmt_dev_s *dev);
|
||||
static int rmt_setup(struct rmt_dev_s *dev);
|
||||
IRAM_ATTR static int rmt_interrupt(int irq, void *context, void *arg);
|
||||
|
||||
/****************************************************************************
|
||||
* Private Data
|
||||
****************************************************************************/
|
||||
|
||||
struct rmt_dev_s g_rmt_dev =
|
||||
{
|
||||
.periph = ESP32_PERIPH_RMT,
|
||||
.irq = ESP32_IRQ_RMT,
|
||||
.cpu = 0,
|
||||
.cpuint = -ENOMEM,
|
||||
.lock = 0
|
||||
};
|
||||
|
||||
/****************************************************************************
|
||||
* Private Functions
|
||||
****************************************************************************/
|
||||
|
||||
/****************************************************************************
|
||||
* Name: rmt_reset
|
||||
*
|
||||
* Description:
|
||||
* Reset the RMT device. Called early to initialize the hardware. This
|
||||
* function is called, before esp32_rmt_setup().
|
||||
*
|
||||
* Input Parameters:
|
||||
* dev - An instance of the "upper half" RMT driver state structure.
|
||||
*
|
||||
* Returned Value:
|
||||
* None
|
||||
*
|
||||
****************************************************************************/
|
||||
|
||||
static void rmt_reset(struct rmt_dev_s *dev)
|
||||
{
|
||||
irqstate_t flags;
|
||||
|
||||
flags = spin_lock_irqsave(&dev->lock);
|
||||
|
||||
modifyreg32(DPORT_PERIP_RST_EN_REG, 0, DPORT_RMT_RST);
|
||||
modifyreg32(DPORT_PERIP_RST_EN_REG, DPORT_RMT_RST, 0);
|
||||
|
||||
/* Clear any spurious IRQ Flag */
|
||||
|
||||
putreg32(0xffffffff, RMT_INT_CLR_REG);
|
||||
|
||||
/* Enable memory wrap-around */
|
||||
|
||||
modifyreg32(RMT_APB_CONF_REG, 0 , BIT(1));
|
||||
|
||||
spin_unlock_irqrestore(&dev->lock, flags);
|
||||
}
|
||||
|
||||
/****************************************************************************
|
||||
* Name: rmt_setup
|
||||
*
|
||||
* Description:
|
||||
* Configure the RMT. This method is called the first time that the RMT
|
||||
* device is opened. This will occur when the port is first opened.
|
||||
* This setup includes configuring and attaching RMT interrupts.
|
||||
*
|
||||
* Input Parameters:
|
||||
* dev - An instance of the "upper half" RMT driver state structure.
|
||||
*
|
||||
* Returned Value:
|
||||
* Zero on success; a negated errno on failure
|
||||
*
|
||||
****************************************************************************/
|
||||
|
||||
static int rmt_setup(struct rmt_dev_s *dev)
|
||||
{
|
||||
irqstate_t flags;
|
||||
int ret = OK;
|
||||
|
||||
flags = spin_lock_irqsave(&dev->lock);
|
||||
|
||||
if (dev->cpuint != -ENOMEM)
|
||||
{
|
||||
/* Disable the provided CPU Interrupt to configure it. */
|
||||
|
||||
up_disable_irq(dev->cpuint);
|
||||
}
|
||||
|
||||
dev->cpu = up_cpu_index();
|
||||
dev->cpuint = esp32_setup_irq(dev->cpu, dev->periph,
|
||||
1, ESP32_CPUINT_LEVEL);
|
||||
if (dev->cpuint < 0)
|
||||
{
|
||||
/* Failed to allocate a CPU interrupt of this type. */
|
||||
|
||||
ret = dev->cpuint;
|
||||
spin_unlock_irqrestore(&dev->lock, flags);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
ret = irq_attach(dev->irq, rmt_interrupt, dev);
|
||||
|
||||
if (ret != OK)
|
||||
{
|
||||
/* Failed to attach IRQ, so CPU interrupt must be freed. */
|
||||
|
||||
esp32_teardown_irq(dev->cpu, dev->periph, dev->cpuint);
|
||||
dev->cpuint = -ENOMEM;
|
||||
spin_unlock_irqrestore(&dev->lock, flags);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
/* Enable the CPU interrupt that is linked to the RMT device. */
|
||||
|
||||
up_enable_irq(dev->irq);
|
||||
|
||||
spin_unlock_irqrestore(&dev->lock, flags);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
/****************************************************************************
|
||||
* Name: rmt_load_tx_buffer
|
||||
*
|
||||
* Description:
|
||||
* Copies chunks of data from the buffer to the RMT device memory
|
||||
* This function can also be called on the first transmission data chunk
|
||||
*
|
||||
* Input Parameters:
|
||||
* channel - Pointer to the channel to be reloaded
|
||||
*
|
||||
* Returned Value:
|
||||
* None
|
||||
*
|
||||
****************************************************************************/
|
||||
|
||||
IRAM_ATTR void rmt_load_tx_buffer(struct rmt_dev_channel_s *channel)
|
||||
{
|
||||
uint32_t *src = channel->src;
|
||||
uint32_t dst_mem;
|
||||
uint32_t buffer_size;
|
||||
|
||||
if (channel->src_offset == 0)
|
||||
{
|
||||
buffer_size = channel->available_words;
|
||||
dst_mem = channel->start_address;
|
||||
channel->next_buffer = 0;
|
||||
}
|
||||
else
|
||||
{
|
||||
buffer_size = channel->reload_thresh;
|
||||
dst_mem = channel->start_address +
|
||||
4*channel->next_buffer*channel->reload_thresh;
|
||||
|
||||
/* only swap buffers after the first call */
|
||||
|
||||
if (channel->next_buffer == 0)
|
||||
{
|
||||
channel->next_buffer = 1;
|
||||
}
|
||||
else
|
||||
{
|
||||
channel->next_buffer = 0;
|
||||
}
|
||||
}
|
||||
|
||||
while (channel->src_offset < channel->words_to_send && buffer_size > 0)
|
||||
{
|
||||
uint32_t word_to_send = *(src + channel->src_offset);
|
||||
putreg32(word_to_send, dst_mem);
|
||||
|
||||
channel->src_offset++;
|
||||
dst_mem += 4;
|
||||
buffer_size--;
|
||||
}
|
||||
|
||||
/* Adding 0x00 on RMT's buffer marks the EOT */
|
||||
|
||||
if (channel->src_offset == channel->words_to_send && buffer_size > 0)
|
||||
{
|
||||
putreg32(0x00, dst_mem);
|
||||
}
|
||||
}
|
||||
|
||||
/****************************************************************************
|
||||
* Name: rmt_interrupt
|
||||
*
|
||||
* Description:
|
||||
* RMT TX interrupt handler
|
||||
*
|
||||
* Input Parameters:
|
||||
* irq - The IRQ number of the interrupt.
|
||||
* context - The register state save array at the time of the interrupt.
|
||||
* arg - The pointer to driver structure.
|
||||
*
|
||||
* Returned Value:
|
||||
* Zero on success; a negated errno on failure
|
||||
*
|
||||
****************************************************************************/
|
||||
|
||||
IRAM_ATTR static int rmt_interrupt(int irq, void *context, void *arg)
|
||||
{
|
||||
struct rmt_dev_s *dev = (struct rmt_dev_s *)arg;
|
||||
uint32_t regval = getreg32(RMT_INT_ST_REG);
|
||||
|
||||
uint8_t error_flag = 0;
|
||||
|
||||
int flags = spin_lock_irqsave(&dev->lock);
|
||||
|
||||
for (int ch_idx = 0; ch_idx < RMT_NUMBER_OF_CHANNELS; ch_idx++)
|
||||
{
|
||||
struct rmt_dev_channel_s *channel_data =
|
||||
(struct rmt_dev_channel_s *) &(dev->channels[ch_idx]);
|
||||
|
||||
/* IRQs from channels with no pins, should be ignored */
|
||||
|
||||
if (channel_data->output_pin < 0)
|
||||
{
|
||||
putreg32(RMT_CHN_TX_THR_EVENT_INT_CLR(ch_idx), RMT_INT_CLR_REG);
|
||||
putreg32(RMT_CHN_TX_END_INT_CLR(ch_idx), RMT_INT_CLR_REG);
|
||||
continue;
|
||||
}
|
||||
|
||||
if (regval & RMT_CHN_TX_THR_EVENT_INT_ST(ch_idx))
|
||||
{
|
||||
putreg32(RMT_CHN_TX_THR_EVENT_INT_CLR(ch_idx), RMT_INT_CLR_REG);
|
||||
|
||||
/* buffer refill */
|
||||
|
||||
rmt_load_tx_buffer(channel_data);
|
||||
}
|
||||
else if (regval & RMT_CHN_TX_END_INT_ST(ch_idx))
|
||||
{
|
||||
/* end of transmission */
|
||||
|
||||
modifyreg32(RMT_INT_ENA_REG,
|
||||
RMT_CHN_TX_END_INT_ENA(ch_idx) |
|
||||
RMT_CHN_TX_THR_EVENT_INT_ENA(ch_idx),
|
||||
0
|
||||
);
|
||||
|
||||
putreg32(RMT_CHN_TX_END_INT_CLR(ch_idx), RMT_INT_CLR_REG);
|
||||
putreg32(RMT_CHN_TX_THR_EVENT_INT_CLR(ch_idx), RMT_INT_CLR_REG);
|
||||
|
||||
/* release the lock so the write function can return */
|
||||
|
||||
nxsem_post(&channel_data->tx_sem);
|
||||
}
|
||||
}
|
||||
|
||||
if (error_flag)
|
||||
{
|
||||
/* clear any spurious IRQ flag */
|
||||
|
||||
putreg32(0xffffffff, RMT_INT_CLR_REG);
|
||||
}
|
||||
|
||||
spin_unlock_irqrestore(&dev->lock, flags);
|
||||
return OK;
|
||||
}
|
||||
|
||||
/****************************************************************************
|
||||
* Public Functions
|
||||
****************************************************************************/
|
||||
|
||||
/****************************************************************************
|
||||
* Name: esp32_rmtinitialize
|
||||
*
|
||||
* Description:
|
||||
* Initialize the selected RMT device
|
||||
*
|
||||
* Input Parameters:
|
||||
*
|
||||
* Returned Value:
|
||||
* Valid RMT device structure reference on success; a NULL on failure
|
||||
*
|
||||
****************************************************************************/
|
||||
|
||||
struct rmt_dev_s *esp32_rmtinitialize(void)
|
||||
{
|
||||
struct rmt_dev_s *rmtdev = &g_rmt_dev;
|
||||
irqstate_t flags;
|
||||
|
||||
flags = spin_lock_irqsave(&rmtdev->lock);
|
||||
|
||||
modifyreg32(DPORT_PERIP_CLK_EN_REG, 0, DPORT_RMT_CLK_EN);
|
||||
modifyreg32(DPORT_PERIP_RST_EN_REG, DPORT_RMT_RST, 0);
|
||||
|
||||
spin_unlock_irqrestore(&rmtdev->lock, flags);
|
||||
|
||||
rmt_reset(rmtdev);
|
||||
rmt_setup(rmtdev);
|
||||
|
||||
rmtdev->channels = kmm_zalloc(sizeof(struct rmt_dev_channel_s) *
|
||||
RMT_NUMBER_OF_CHANNELS);
|
||||
if (!rmtdev->channels)
|
||||
{
|
||||
rmterr("Failed to allocate memory for RMT Channels");
|
||||
return NULL;
|
||||
}
|
||||
|
||||
for (int ch_idx = 0; ch_idx < RMT_NUMBER_OF_CHANNELS; ch_idx++)
|
||||
{
|
||||
struct rmt_dev_channel_s *channel_data =
|
||||
(struct rmt_dev_channel_s *) &(rmtdev->channels[ch_idx]);
|
||||
|
||||
channel_data->open_count = 0;
|
||||
channel_data->ch_idx = ch_idx;
|
||||
channel_data->output_pin = -1;
|
||||
|
||||
channel_data->available_words = 64;
|
||||
uint32_t start_addr_chn = RMT_DATA_BASE_ADDR +
|
||||
RMT_DATA_MEMORY_BLOCK_WORDS * 4 * ch_idx;
|
||||
|
||||
channel_data->start_address = start_addr_chn;
|
||||
channel_data->reload_thresh = channel_data->available_words / 2;
|
||||
channel_data->parent_dev = rmtdev;
|
||||
}
|
||||
|
||||
return rmtdev;
|
||||
}
|
||||
|
||||
/****************************************************************************
|
||||
* Name: rmt_attach_pin_to_channel
|
||||
*
|
||||
* Description:
|
||||
* Binds a gpio pin to a RMT channel
|
||||
*
|
||||
* Input Parameters:
|
||||
* rmtdev - pointer the rmt device, needed for the locks
|
||||
* output_pin - the pin used for output
|
||||
* channel - the RMT's channel that will be used
|
||||
*
|
||||
* Returned Value:
|
||||
* Zero on success; a negated errno on failure
|
||||
*
|
||||
****************************************************************************/
|
||||
|
||||
int rmt_attach_pin_to_channel(struct rmt_dev_s *rmtdev, int ch_idx, int pin)
|
||||
{
|
||||
irqstate_t flags;
|
||||
|
||||
if (ch_idx >= RMT_NUMBER_OF_CHANNELS || pin < 0)
|
||||
{
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
flags = spin_lock_irqsave(&rmtdev->lock);
|
||||
|
||||
struct rmt_dev_channel_s *channel_data =
|
||||
(struct rmt_dev_channel_s *) &(rmtdev->channels[ch_idx]);
|
||||
|
||||
channel_data->output_pin = pin;
|
||||
nxsem_init(&channel_data->tx_sem, 0, 1);
|
||||
|
||||
/* Configure RMT GPIO pin */
|
||||
|
||||
esp32_gpio_matrix_out(pin, RMT_SIG_OUT0_IDX + ch_idx, 0, 0);
|
||||
esp32_configgpio(pin, OUTPUT_FUNCTION_1);
|
||||
|
||||
spin_unlock_irqrestore(&rmtdev->lock, flags);
|
||||
|
||||
return OK;
|
||||
}
|
||||
|
||||
#endif
|
@ -1,164 +0,0 @@
|
||||
/****************************************************************************
|
||||
* arch/xtensa/src/esp32/esp32_rmt.h
|
||||
*
|
||||
* 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.
|
||||
*
|
||||
****************************************************************************/
|
||||
|
||||
#ifndef __ARCH_XTENSA_SRC_ESP32_ESP32_RMT_H
|
||||
#define __ARCH_XTENSA_SRC_ESP32_ESP32_RMT_H
|
||||
|
||||
/****************************************************************************
|
||||
* Included Files
|
||||
****************************************************************************/
|
||||
|
||||
#include <nuttx/config.h>
|
||||
#include <semaphore.h>
|
||||
#include <nuttx/spinlock.h>
|
||||
#include "hardware/esp32_rmt.h"
|
||||
|
||||
/****************************************************************************
|
||||
* Pre-processor Definitions
|
||||
****************************************************************************/
|
||||
|
||||
/****************************************************************************
|
||||
* Public Types
|
||||
****************************************************************************/
|
||||
|
||||
struct rmt_dev_channel_s
|
||||
{
|
||||
/* Parameters for each RMT channel */
|
||||
|
||||
int open_count; /* Unused */
|
||||
int ch_idx; /* RMT channel number (0-7) */
|
||||
int output_pin; /* GPIO pin number attached to the RMT */
|
||||
int next_buffer; /* Tracking buffer (0 or 1) used next reload */
|
||||
sem_t tx_sem; /* Semaphore structure */
|
||||
|
||||
uint32_t *src; /* Data to be copied to the internal buffer */
|
||||
uint32_t src_offset; /* Offset pointer to the src */
|
||||
size_t words_to_send; /* Number of 32-bit words to be sent */
|
||||
uint32_t available_words; /* Counter of available words in the src */
|
||||
uint32_t start_address; /* Current RMT register buffer address */
|
||||
uint32_t reload_thresh; /* Threshold for reloading the internal buffer */
|
||||
void *parent_dev; /* Pointer to the parent RMT device structure */
|
||||
};
|
||||
|
||||
struct rmt_dev_s
|
||||
{
|
||||
/* Device configuration */
|
||||
|
||||
uint8_t periph; /* Peripheral ID */
|
||||
uint8_t irq; /* IRQ associated with this RMT */
|
||||
uint8_t cpu; /* CPU ID */
|
||||
int cpuint; /* CPU interrupt assigned to this RMT */
|
||||
spinlock_t lock;
|
||||
|
||||
struct rmt_dev_channel_s *channels;
|
||||
};
|
||||
|
||||
/****************************************************************************
|
||||
* Public Data
|
||||
****************************************************************************/
|
||||
|
||||
#ifndef __ASSEMBLY__
|
||||
#ifdef __cplusplus
|
||||
extern "C"
|
||||
{
|
||||
#endif
|
||||
|
||||
/****************************************************************************
|
||||
* Public Functions Prototypes
|
||||
****************************************************************************/
|
||||
|
||||
#if defined(CONFIG_ESP32_RMT)
|
||||
|
||||
/****************************************************************************
|
||||
* Name: rmt_load_tx_buffer
|
||||
*
|
||||
* Description:
|
||||
* Copies chunks of data from the buffer to the RMT device memory
|
||||
* This function can also be called on the first transmission data chunk
|
||||
*
|
||||
* Input Parameters:
|
||||
* channel - Pointer to the channel to be reloaded
|
||||
*
|
||||
* Returned Value:
|
||||
* None
|
||||
*
|
||||
****************************************************************************/
|
||||
|
||||
IRAM_ATTR void rmt_load_tx_buffer(struct rmt_dev_channel_s *channel);
|
||||
|
||||
/****************************************************************************
|
||||
* Name: esp32_rmtinitialize
|
||||
*
|
||||
* Description:
|
||||
* Initialize the selected RMT device
|
||||
*
|
||||
* Input Parameters:
|
||||
*
|
||||
* Returned Value:
|
||||
* Valid RMT device structure reference on success; a NULL on failure
|
||||
*
|
||||
****************************************************************************/
|
||||
|
||||
struct rmt_dev_s *esp32_rmtinitialize(void);
|
||||
|
||||
/****************************************************************************
|
||||
* Name: rmt_attach_pin_to_channel
|
||||
*
|
||||
* Description:
|
||||
* Binds a gpio pin to a RMT channel
|
||||
*
|
||||
* Input Parameters:
|
||||
* rmtdev - pointer the rmt device, needed for the locks
|
||||
* output_pin - the pin used for output
|
||||
* channel - the RMT's channel that will be used
|
||||
*
|
||||
* Returned Value:
|
||||
* Zero on success; a negated errno on failure
|
||||
*
|
||||
****************************************************************************/
|
||||
|
||||
int rmt_attach_pin_to_channel(struct rmt_dev_s *rmtdev, int ch_idx, int pin);
|
||||
|
||||
/****************************************************************************
|
||||
* Name: board_rmt_initialize
|
||||
*
|
||||
* Description:
|
||||
* Initialize RMT driver and register the channel/pin pair at /dev/rtm0
|
||||
*
|
||||
* Input Parameters:
|
||||
* output_pin - the output pin to assing to the channel
|
||||
* channel - the channel that will be initialized
|
||||
*
|
||||
* Returned Value:
|
||||
* Zero (OK) is returned on success; A negated errno value is returned
|
||||
* to indicate the nature of any failure.
|
||||
*
|
||||
****************************************************************************/
|
||||
|
||||
int board_rmt_initialize(int output_pin, int channel);
|
||||
|
||||
#endif
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
||||
#endif /* __ASSEMBLY__ */
|
||||
|
||||
#endif /* __ARCH_XTENSA_SRC_ESP32_ESP32_RMT_H */
|
@ -1,266 +0,0 @@
|
||||
/****************************************************************************
|
||||
* arch/xtensa/src/esp32/hardware/esp32_rmt.h
|
||||
*
|
||||
* 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.
|
||||
*
|
||||
****************************************************************************/
|
||||
|
||||
#ifndef __ARCH_XTENSA_SRC_ESP32_HARDWARE_ESP32_RMT_H
|
||||
#define __ARCH_XTENSA_SRC_ESP32_HARDWARE_ESP32_RMT_H
|
||||
|
||||
/****************************************************************************
|
||||
* Included Files
|
||||
****************************************************************************/
|
||||
|
||||
#include "esp32_soc.h"
|
||||
|
||||
/****************************************************************************
|
||||
* Pre-processor Definitions
|
||||
****************************************************************************/
|
||||
|
||||
/* RMT Peripheral constants */
|
||||
#define RMT_NUMBER_OF_CHANNELS 8
|
||||
#define RMT_DATA_BASE_ADDR 0x3ff56800
|
||||
#define RMT_DATA_MEMORY_BLOCK_WORDS 64
|
||||
|
||||
/* RMT Channel configuration registers */
|
||||
|
||||
#define RMT_CHNCONF_REG_BASE (DR_REG_RMT_BASE+0x20)
|
||||
#define RMT_CHNCONF0_REG(n) (RMT_CHNCONF_REG_BASE + 8*n)
|
||||
#define RMT_CHNCONF1_REG(n) (RMT_CHNCONF0_REG(n) + 4)
|
||||
|
||||
#define RMT_INT_RAW_REG (DR_REG_RMT_BASE+0x00A0)
|
||||
#define RMT_INT_ST_REG (DR_REG_RMT_BASE+0x00A4)
|
||||
#define RMT_INT_ENA_REG (DR_REG_RMT_BASE+0x00A8)
|
||||
#define RMT_INT_CLR_REG (DR_REG_RMT_BASE+0x00AC)
|
||||
|
||||
#define RMT_CHNCARRIER_DUTY_REG(n) (DR_REG_RMT_BASE + 0x00B0+4*n)
|
||||
#define RMT_CHN_TX_LIM_REG(n) (DR_REG_RMT_BASE + 0x00D0+4*n)
|
||||
|
||||
#define RMT_APB_CONF_REG (DR_REG_RMT_BASE + 0x00F0)
|
||||
|
||||
/* RMT_CHNCONF0_REG Bits */
|
||||
|
||||
/* RMT_MEM_PD: This bit is used to power down the entire RMT RAM block.
|
||||
* (It only exists in RMT_CH0CONF0).
|
||||
* 1: power down memory; 0: power up memory. (R/W)
|
||||
*/
|
||||
|
||||
#define RMT_MEM_PD BIT(30)
|
||||
#define RMT_MEM_PD_M (RMT_MEM_PD_V << RMT_MEM_PD_S)
|
||||
#define RMT_MEM_PD_V 0x00000001
|
||||
#define RMT_MEM_PD_S 30
|
||||
|
||||
/* RMT_CARRIER_OUT_LV_CHN This bit is used for configuration when the
|
||||
* carrier wave is being transmitted. Transmit on low output level with 0,
|
||||
* and transmit on high output level with 1. (R/W)
|
||||
*/
|
||||
|
||||
#define RMT_CARRIER_OUT_LV_CHN BIT(29)
|
||||
#define RMT_CARRIER_OUT_LV_CHN_M (RMT_CARRIER_OUT_LV_CHN_V << RMT_CARRIER_OUT_LV_CHN_S)
|
||||
#define RMT_CARRIER_OUT_LV_CHN_V 0x00000001
|
||||
#define RMT_CARRIER_OUT_LV_CHN_S 29
|
||||
|
||||
/* RMT_CARRIER_EN_CHN This is the carrier modulation enable-control bit
|
||||
* for channel n. Carrier modulation is enabled with 1, while carrier
|
||||
* modulation is disabled with 0. (R/W)
|
||||
*/
|
||||
|
||||
#define RMT_CARRIER_EN_CHN BIT(28)
|
||||
#define RMT_CARRIER_EN_CHN_M (RMT_CARRIER_EN_CHN_V << RMT_CARRIER_EN_CHN_S)
|
||||
#define RMT_CARRIER_EN_CHN_V 0x00000001
|
||||
#define RMT_CARRIER_EN_CHN_S 28
|
||||
|
||||
/* RMT_MEM_SIZE_CHN This register is used to configure the amount of
|
||||
* memory blocks allocated to channel n. (R/W)
|
||||
*/
|
||||
|
||||
#define RMT_MEM_SIZE_CHN BIT(24)
|
||||
#define RMT_MEM_SIZE_CHN_M (RMT_MEM_SIZE_CHN_V << RMT_MEM_SIZE_CHN_S)
|
||||
#define RMT_MEM_SIZE_CHN_V 0x00000001
|
||||
#define RMT_MEM_SIZE_CHN_S 24
|
||||
|
||||
/* RMT_IDLE_THRES_CHN In receive mode, when no edge is detected on
|
||||
* the input signal for longer than REG_IDLE_THRES_CHN channel clock cycles,
|
||||
* the receive process is finished. (R/W)
|
||||
*/
|
||||
|
||||
#define RMT_IDLE_THRES_CHN BIT(8)
|
||||
#define RMT_IDLE_THRES_CHN_M (RMT_IDLE_THRES_CHN_V << RMT_IDLE_THRES_CHN_S)
|
||||
#define RMT_IDLE_THRES_CHN_V 0x00000001
|
||||
#define RMT_IDLE_THRES_CHN_S 8
|
||||
|
||||
/* RMT_DIV_CNT_CHN This register is used to set the divider for the channel
|
||||
* clock of channel n. (R/W)
|
||||
*/
|
||||
|
||||
#define RMT_DIV_CNT_CHN BIT(0)
|
||||
#define RMT_DIV_CNT_CHN_M (RMT_DIV_CNT_CHN_V << RMT_DIV_CNT_CHN_S)
|
||||
#define RMT_DIV_CNT_CHN_V 0x00000001
|
||||
#define RMT_DIV_CNT_CHN_S 0
|
||||
|
||||
/* RMT_CHNCONF1_REG Bits */
|
||||
|
||||
/* RMT_IDLE_OUT_EN_CHN This is the output enable-control bit for channel n
|
||||
* in IDLE state. (R/W)
|
||||
*/
|
||||
|
||||
#define RMT_IDLE_OUT_EN_CHN BIT(19)
|
||||
#define RMT_IDLE_OUT_EN_CHN_M (RMT_IDLE_OUT_EN_CHN_V << RMT_IDLE_OUT_EN_CHN_S)
|
||||
#define RMT_IDLE_OUT_EN_CHN_V 0x00000001
|
||||
#define RMT_IDLE_OUT_EN_CHN_S 19
|
||||
|
||||
/* RMT_IDLE_OUT_LV_CHN This bit configures the level of output signals
|
||||
* in channel n when the latter is in IDLE state. (R/W)
|
||||
*/
|
||||
|
||||
#define RMT_IDLE_OUT_LV_CHN BIT(18)
|
||||
#define RMT_IDLE_OUT_LV_CHN_M (RMT_IDLE_OUT_LV_CHN_V << RMT_IDLE_OUT_LV_CHN_S)
|
||||
#define RMT_IDLE_OUT_LV_CHN_V 0x00000001
|
||||
#define RMT_IDLE_OUT_LV_CHN_S 18
|
||||
|
||||
/* RMT_REF_ALWAYS_ON_CHN This bit is used to select the channel's base
|
||||
* clock. 1:clk_apb; 0:clk_ref. (R/W)
|
||||
*/
|
||||
|
||||
#define RMT_REF_ALWAYS_ON_CHN BIT(17)
|
||||
#define RMT_REF_ALWAYS_ON_CHN_M (RMT_REF_ALWAYS_ON_CHN_V << RMT_REF_ALWAYS_ON_CHN_S)
|
||||
#define RMT_REF_ALWAYS_ON_CHN_V 0x00000001
|
||||
#define RMT_REF_ALWAYS_ON_CHN_S 17
|
||||
|
||||
/* RMT_REF_CNT_RST_CHN Setting this bit resets the clock divider of channel
|
||||
* n. (R/W)
|
||||
*/
|
||||
|
||||
#define RMT_REF_CNT_RST_CHN BIT(16)
|
||||
#define RMT_REF_CNT_RST_CHN_M (RMT_REF_CNT_RST_CHN_V << RMT_REF_CNT_RST_CHN_S)
|
||||
#define RMT_REF_CNT_RST_CHN_V 0x00000001
|
||||
#define RMT_REF_CNT_RST_CHN_S 16
|
||||
|
||||
/* RMT_RX_FILTER_THRES_CHN In receive mode, channel n ignores input
|
||||
* pulse when the pulse width is smaller than this value in APB clock
|
||||
* periods. (R/W)
|
||||
*/
|
||||
|
||||
#define RMT_RX_FILTER_THRES_CHN BIT(8)
|
||||
#define RMT_RX_FILTER_THRES_CHN_M (RMT_RX_FILTER_THRES_CHN_V << RMT_RX_FILTER_THRES_CHN_S)
|
||||
#define RMT_RX_FILTER_THRES_CHN_V 0x00000001
|
||||
#define RMT_RX_FILTER_THRES_CHN_S 8
|
||||
|
||||
/* RMT_RX_FILTER_EN_CHN This is the receive filter's enable-bit for channel
|
||||
* n. (R/W)
|
||||
*/
|
||||
|
||||
#define RMT_RX_FILTER_EN_CHN BIT(7)
|
||||
#define RMT_RX_FILTER_EN_CHN_M (RMT_RX_FILTER_EN_CHN_V << RMT_RX_FILTER_EN_CHN_S)
|
||||
#define RMT_RX_FILTER_EN_CHN_V 0x00000001
|
||||
#define RMT_RX_FILTER_EN_CHN_S 7
|
||||
|
||||
/* RMT_TX_CONTI_MODE_CHN If this bit is set, instead of going to an idle
|
||||
* state when transmission ends, the transmitter will restart transmission.
|
||||
* This results in a repeating output signal. (R/W)
|
||||
*/
|
||||
|
||||
#define RMT_TX_CONTI_MODE_CHN BIT(6)
|
||||
#define RMT_TX_CONTI_MODE_CHN_M (RMT_TX_CONTI_MODE_CHN_V << RMT_TX_CONTI_MODE_CHN_S)
|
||||
#define RMT_TX_CONTI_MODE_CHN_V 0x00000001
|
||||
#define RMT_TX_CONTI_MODE_CHN_S 6
|
||||
|
||||
/* RMT_MEM_OWNER_CHN This bit marks channel n's RAM block ownership.
|
||||
* Number 1 indicates that the receiver is using the RAM, while 0 indicates
|
||||
* that the transmitter is using the RAM. (R/W)
|
||||
*/
|
||||
|
||||
#define RMT_MEM_OWNER_CHN BIT(5)
|
||||
#define RMT_MEM_OWNER_CHN_M (RMT_MEM_OWNER_CHN_V << RMT_MEM_OWNER_CHN_S)
|
||||
#define RMT_MEM_OWNER_CHN_V 0x00000001
|
||||
#define RMT_MEM_OWNER_CHN_S 5
|
||||
|
||||
/* RMT_MEM_RD_RST_CHN Set this bit to reset the read-RAM address for channel
|
||||
* n by accessing the transmitter. (R/W)
|
||||
*/
|
||||
|
||||
#define RMT_MEM_RD_RST_CHN BIT(3)
|
||||
#define RMT_MEM_RD_RST_CHN_M (RMT_MEM_RD_RST_CHN_V << RMT_MEM_RD_RST_CHN_S)
|
||||
#define RMT_MEM_RD_RST_CHN_V 0x00000001
|
||||
#define RMT_MEM_RD_RST_CHN_S 3
|
||||
|
||||
/* RMT_MEM_WR_RST_CHN Set this bit to reset the write-RAM address for
|
||||
* channel n by accessing the receiver. (R/W)
|
||||
*/
|
||||
|
||||
#define RMT_MEM_WR_RST_CHN BIT(2)
|
||||
#define RMT_MEM_WR_RST_CHN_M (RMT_MEM_WR_RST_CHN_V << RMT_MEM_WR_RST_CHN_S)
|
||||
#define RMT_MEM_WR_RST_CHN_V 0x00000001
|
||||
#define RMT_MEM_WR_RST_CHN_S 2
|
||||
|
||||
/* RMT_RX_EN_CHN Set this bit to enable receiving data on channel n. (R/W) */
|
||||
|
||||
#define RMT_RX_EN_CHN BIT(1)
|
||||
#define RMT_RX_EN_CHN_M (RMT_RX_EN_CHN_V << RMT_RX_EN_CHN_S)
|
||||
#define RMT_RX_EN_CHN_V 0x00000001
|
||||
#define RMT_RX_EN_CHN_S 1
|
||||
|
||||
/* RMT_TX_START_CHN Set this bit to start sending data on channel n. (R/W) */
|
||||
|
||||
#define RMT_TX_START_CHN(n) BIT(n)
|
||||
#define RMT_TX_START_CHN_M (RMT_TX_START_CHN_V << RMT_TX_START_CHN_S)
|
||||
#define RMT_TX_START_CHN_V 0x00000001
|
||||
#define RMT_TX_START_CHN_S 0
|
||||
|
||||
/* RMT_INT_RAW_REG Bits */
|
||||
|
||||
/* RMT_CHN_TX_THR_EVENT_INT_RAW The raw interrupt status bit for the
|
||||
* RMT_CHN_TX_THR_EVENT_INT interrupt. (RO)
|
||||
*/
|
||||
|
||||
#define RMT_CHN_TX_THR_EVENT_INT_RAW(n) BIT(24+n)
|
||||
|
||||
/* RMT_CHN_ERR_INT_RAW The raw interrupt status bit for the RMT_CHN_ERR_INT
|
||||
* interrupt. (RO)
|
||||
*/
|
||||
|
||||
#define RMT_CHN_ERR_INT_RAW(n) BIT(3*n+2)
|
||||
|
||||
/* RMT_CHN_RX_END_INT_RAW The raw interrupt status bit for
|
||||
* the RMT_CHN_RX_END_INT interrupt. (RO)
|
||||
*/
|
||||
#define RMT_CHN_RX_END_INT_RAW(n) BIT(3*n+1)
|
||||
|
||||
/* RMT_CHN_TX_END_INT_RAW The raw interrupt status bit for the
|
||||
* RMT_CHN_TX_END_INT interrupt. (RO)
|
||||
*/
|
||||
#define RMT_CHN_TX_END_INT_RAW(n) BIT(3*n)
|
||||
|
||||
/* RMT_INT_ST_REG Bits */
|
||||
#define RMT_CHN_TX_THR_EVENT_INT_ST(n) BIT(24+n)
|
||||
#define RMT_CHN_ERR_INT_ST(n) BIT(3*n+2)
|
||||
#define RMT_CHN_RX_END_INT_ST(n) BIT(3*n+1)
|
||||
#define RMT_CHN_TX_END_INT_ST(n) BIT(3*n)
|
||||
|
||||
/* RMT_INT_ENA_REG Bits */
|
||||
#define RMT_CHN_TX_THR_EVENT_INT_ENA(n) BIT(24+n)
|
||||
#define RMT_CHN_ERR_INT_ENA(n) BIT(3*n+2)
|
||||
#define RMT_CHN_RX_END_INT_ENA(n) BIT(3*n+1)
|
||||
#define RMT_CHN_TX_END_INT_ENA(n) BIT(3*n)
|
||||
|
||||
/* RMT_INT_CLR_REG Bits */
|
||||
#define RMT_CHN_TX_THR_EVENT_INT_CLR(n) BIT(24+n)
|
||||
#define RMT_CHN_ERR_INT_CLR(n) BIT(3*n+2)
|
||||
#define RMT_CHN_RX_END_INT_CLR(n) BIT(3*n+1)
|
||||
#define RMT_CHN_TX_END_INT_CLR(n) BIT(3*n)
|
||||
|
||||
#endif /* __ARCH_XTENSA_SRC_ESP32_HARDWARE_ESP32_RMT_H */
|
@ -1,282 +0,0 @@
|
||||
/****************************************************************************
|
||||
* boards/xtensa/esp32/common/src/esp32_rmt.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 <errno.h>
|
||||
#include <debug.h>
|
||||
#include <stdio.h>
|
||||
|
||||
#include "xtensa.h"
|
||||
|
||||
#include <nuttx/kmalloc.h>
|
||||
#include "esp32_rmt.h"
|
||||
|
||||
#ifdef CONFIG_ESP32_RMT
|
||||
|
||||
/****************************************************************************
|
||||
* Pre-processor Definitions
|
||||
****************************************************************************/
|
||||
|
||||
#define APB_PERIOD (12.5)
|
||||
|
||||
#define T0H ((uint16_t)(350 / APB_PERIOD)) // ns
|
||||
#define T0L ((uint16_t)(900 / APB_PERIOD)) // ns
|
||||
#define T1H ((uint16_t)(900 / APB_PERIOD)) // ns
|
||||
#define T1L ((uint16_t)(350 / APB_PERIOD)) // ns
|
||||
#define RES ((uint16_t)(60000 / APB_PERIOD)) // ns
|
||||
|
||||
/****************************************************************************
|
||||
* Private Types
|
||||
****************************************************************************/
|
||||
|
||||
/****************************************************************************
|
||||
* Private Function Prototypes
|
||||
****************************************************************************/
|
||||
|
||||
/****************************************************************************
|
||||
* Private Data
|
||||
****************************************************************************/
|
||||
|
||||
/****************************************************************************
|
||||
* Public Data
|
||||
****************************************************************************/
|
||||
|
||||
/****************************************************************************
|
||||
* Private Functions
|
||||
****************************************************************************/
|
||||
|
||||
/****************************************************************************
|
||||
* Public Functions
|
||||
****************************************************************************/
|
||||
|
||||
static int rmt_open(struct file *filep)
|
||||
{
|
||||
struct inode *inode = filep->f_inode;
|
||||
struct rmt_dev_channel_s *dev_data = inode->i_private;
|
||||
|
||||
struct rmt_dev_s *parent_dev =
|
||||
(struct rmt_dev_s *)dev_data->parent_dev;
|
||||
int ret;
|
||||
irqstate_t flags;
|
||||
DEBUGASSERT(parent_dev);
|
||||
|
||||
nxsem_wait(&dev_data->tx_sem);
|
||||
|
||||
flags = spin_lock_irqsave(&parent_dev->lock);
|
||||
|
||||
if (dev_data->open_count == 0)
|
||||
{
|
||||
int ch_idx = dev_data->ch_idx;
|
||||
|
||||
uint32_t reg0_addr = RMT_CHNCONF0_REG(ch_idx);
|
||||
uint32_t reg1_addr = RMT_CHNCONF1_REG(ch_idx);
|
||||
uint32_t reg_val = 0x00;
|
||||
|
||||
/* a single memory block with double buffering is enough */
|
||||
|
||||
uint32_t mem_blocks = 1;
|
||||
dev_data->available_words = RMT_DATA_MEMORY_BLOCK_WORDS*mem_blocks;
|
||||
dev_data->reload_thresh = dev_data->available_words / 2;
|
||||
uint32_t start_addr_chn = RMT_DATA_BASE_ADDR +
|
||||
RMT_DATA_MEMORY_BLOCK_WORDS * 4 * ch_idx;
|
||||
|
||||
dev_data->start_address = start_addr_chn;
|
||||
|
||||
reg_val = (mem_blocks) << 24;
|
||||
uint32_t clock_divider = 1;
|
||||
reg_val |= (clock_divider);
|
||||
putreg32(reg_val, reg0_addr);
|
||||
reg_val = 0;
|
||||
|
||||
/* use APB clock */
|
||||
|
||||
reg_val |= RMT_REF_ALWAYS_ON_CHN;
|
||||
|
||||
/* memory block in transmission mode */
|
||||
|
||||
reg_val &= ~RMT_MEM_OWNER_CHN;
|
||||
putreg32(reg_val, reg1_addr);
|
||||
|
||||
/* set when the buffer swapping IRQ must be generated */
|
||||
|
||||
uint32_t reload_addr = RMT_CHN_TX_LIM_REG(ch_idx);
|
||||
rmtinfo("Setting thr limit at %08X to %d",
|
||||
reload_addr, dev_data->reload_thresh);
|
||||
putreg32(dev_data->reload_thresh, reload_addr);
|
||||
|
||||
/* allow direct access to RMT's memory */
|
||||
|
||||
modifyreg32(RMT_APB_CONF_REG, 0, BIT(0));
|
||||
}
|
||||
else
|
||||
{
|
||||
rmtwarn("Be careful on opening this channel multiple times");
|
||||
}
|
||||
|
||||
dev_data->open_count += 1;
|
||||
|
||||
ret = OK;
|
||||
|
||||
spin_unlock_irqrestore(&parent_dev->lock, flags);
|
||||
nxsem_post(&dev_data->tx_sem);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int rmt_close(struct file *filep)
|
||||
{
|
||||
struct inode *inode = filep->f_inode;
|
||||
struct rmt_dev_channel_s *dev_data = inode->i_private;
|
||||
|
||||
struct rmt_dev_s *parent_dev =
|
||||
(struct rmt_dev_s *)dev_data->parent_dev;
|
||||
|
||||
int ret;
|
||||
irqstate_t flags;
|
||||
DEBUGASSERT(parent_dev);
|
||||
nxsem_wait(&dev_data->tx_sem);
|
||||
flags = spin_lock_irqsave(&parent_dev->lock);
|
||||
|
||||
dev_data->open_count -= 1;
|
||||
|
||||
ret = OK;
|
||||
|
||||
spin_unlock_irqrestore(&parent_dev->lock, flags);
|
||||
nxsem_post(&dev_data->tx_sem);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static ssize_t rmt_write(struct file *filep,
|
||||
const char *data,
|
||||
size_t len)
|
||||
{
|
||||
struct inode *inode = filep->f_inode;
|
||||
struct rmt_dev_channel_s *dev_data = inode->i_private;
|
||||
|
||||
struct rmt_dev_s *parent_dev =
|
||||
(struct rmt_dev_s *)dev_data->parent_dev;
|
||||
|
||||
irqstate_t flags;
|
||||
size_t len_in_words = len / 4;
|
||||
|
||||
DEBUGASSERT(parent_dev);
|
||||
|
||||
if (data == NULL || (len_in_words == 0) || (len % 4))
|
||||
{
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
flags = spin_lock_irqsave(&parent_dev->lock);
|
||||
|
||||
/* set RMT's memory as writable */
|
||||
|
||||
uint32_t reg1_addr = RMT_CHNCONF1_REG(dev_data->ch_idx);
|
||||
modifyreg32(reg1_addr, 0, RMT_MEM_RD_RST_CHN);
|
||||
modifyreg32(reg1_addr, RMT_MEM_RD_RST_CHN, 0);
|
||||
|
||||
dev_data->src = (uint32_t *)data;
|
||||
dev_data->src_offset = 0;
|
||||
dev_data->words_to_send = len_in_words;
|
||||
|
||||
/* enable IRQs for buffer refill and End-of-Transmition (EOT) */
|
||||
|
||||
modifyreg32(
|
||||
RMT_INT_ENA_REG,
|
||||
0,
|
||||
RMT_CHN_TX_THR_EVENT_INT_ENA(dev_data->ch_idx) |
|
||||
RMT_CHN_TX_END_INT_ENA(dev_data->ch_idx));
|
||||
|
||||
rmt_load_tx_buffer(dev_data);
|
||||
|
||||
/* tell RMT to start the transmition */
|
||||
|
||||
modifyreg32(reg1_addr, 0, RMT_TX_START_CHN(dev_data->ch_idx));
|
||||
|
||||
spin_unlock_irqrestore(&parent_dev->lock, flags);
|
||||
|
||||
/* wait for the transmition to finish */
|
||||
|
||||
nxsem_wait(&dev_data->tx_sem);
|
||||
nxsem_post(&dev_data->tx_sem);
|
||||
|
||||
return len;
|
||||
}
|
||||
|
||||
/****************************************************************************
|
||||
* Name: board_rmt_initialize
|
||||
*
|
||||
* Description:
|
||||
* Initialize and register the RMT driver
|
||||
*
|
||||
* Input Parameters:
|
||||
* devno - The device number, used to build the device path as /dev/rmtN
|
||||
* rmt_dev - Pointer to the RMT device that will be used
|
||||
* nleds - number of LEDs
|
||||
*
|
||||
* Returned Value:
|
||||
* Zero (OK) on success; a negated errno value on failure.
|
||||
*
|
||||
****************************************************************************/
|
||||
|
||||
static const struct file_operations g_rmt_channel_fops =
|
||||
{
|
||||
rmt_open, /* open */
|
||||
rmt_close, /* close */
|
||||
NULL, /* read */
|
||||
rmt_write, /* write */
|
||||
NULL, /* seek */
|
||||
NULL, /* ioctl */
|
||||
};
|
||||
|
||||
int board_rmt_initialize(int channel, int output_pin)
|
||||
{
|
||||
struct rmt_dev_s *rmt_dev = esp32_rmtinitialize();
|
||||
DEBUGASSERT(rmt_dev);
|
||||
|
||||
char devpath[13];
|
||||
int ret;
|
||||
|
||||
rmt_attach_pin_to_channel(rmt_dev, channel, output_pin);
|
||||
|
||||
struct rmt_dev_channel_s *channel_data = &(rmt_dev->channels[channel]);
|
||||
|
||||
/* Register the RMT driver at the specified location. */
|
||||
|
||||
snprintf(devpath, sizeof(devpath), "/dev/rmt%d", channel);
|
||||
|
||||
/* Register the character driver */
|
||||
|
||||
ret = register_driver(devpath, &g_rmt_channel_fops, 0666, channel_data);
|
||||
|
||||
if (ret < 0)
|
||||
{
|
||||
rmterr("ERROR: board_rmt_initialize(%s) failed: %d\n",
|
||||
devpath, ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
return OK;
|
||||
}
|
||||
#endif
|
@ -1,49 +0,0 @@
|
||||
#
|
||||
# This file is autogenerated: PLEASE DO NOT EDIT IT.
|
||||
#
|
||||
# You can use "make menuconfig" to make any modifications to the installed .config file.
|
||||
# You can then do "make savedefconfig" to generate a new defconfig file that includes your
|
||||
# modifications.
|
||||
#
|
||||
# CONFIG_ARCH_LEDS is not set
|
||||
# CONFIG_NSH_ARGCAT is not set
|
||||
# CONFIG_NSH_CMDOPT_HEXDUMP is not set
|
||||
CONFIG_ARCH="xtensa"
|
||||
CONFIG_ARCH_BOARD="esp32-devkitc"
|
||||
CONFIG_ARCH_BOARD_COMMON=y
|
||||
CONFIG_ARCH_BOARD_ESP32_DEVKITC=y
|
||||
CONFIG_ARCH_CHIP="esp32"
|
||||
CONFIG_ARCH_CHIP_ESP32=y
|
||||
CONFIG_ARCH_CHIP_ESP32WROVER=y
|
||||
CONFIG_ARCH_STACKDUMP=y
|
||||
CONFIG_ARCH_XTENSA=y
|
||||
CONFIG_BOARD_LOOPSPERMSEC=16717
|
||||
CONFIG_BUILTIN=y
|
||||
CONFIG_ESP32_RMT=y
|
||||
CONFIG_ESP32_UART0=y
|
||||
CONFIG_EXAMPLES_WS2812_ESP32_RMT=y
|
||||
CONFIG_EXPERIMENTAL=y
|
||||
CONFIG_FS_PROCFS=y
|
||||
CONFIG_HAVE_CXX=y
|
||||
CONFIG_HAVE_CXXINITIALIZE=y
|
||||
CONFIG_IDLETHREAD_STACKSIZE=3072
|
||||
CONFIG_INIT_ENTRYPOINT="nsh_main"
|
||||
CONFIG_INIT_STACKSIZE=3072
|
||||
CONFIG_INTELHEX_BINARY=y
|
||||
CONFIG_MM_REGIONS=3
|
||||
CONFIG_NSH_ARCHINIT=y
|
||||
CONFIG_NSH_BUILTIN_APPS=y
|
||||
CONFIG_NSH_FILEIOSIZE=512
|
||||
CONFIG_NSH_LINELEN=64
|
||||
CONFIG_NSH_READLINE=y
|
||||
CONFIG_PREALLOC_TIMERS=4
|
||||
CONFIG_RAM_SIZE=114688
|
||||
CONFIG_RAM_START=0x20000000
|
||||
CONFIG_RR_INTERVAL=200
|
||||
CONFIG_SCHED_WAITPID=y
|
||||
CONFIG_START_DAY=6
|
||||
CONFIG_START_MONTH=12
|
||||
CONFIG_START_YEAR=2011
|
||||
CONFIG_SYSLOG_BUFFER=y
|
||||
CONFIG_SYSTEM_NSH=y
|
||||
CONFIG_UART0_SERIAL_CONSOLE=y
|
@ -161,10 +161,6 @@
|
||||
# include "esp32_max6675.h"
|
||||
#endif
|
||||
|
||||
#ifdef CONFIG_ESP32_RMT
|
||||
# include "esp32_rmt.h"
|
||||
#endif
|
||||
|
||||
#ifdef CONFIG_DAC
|
||||
# include "esp32_board_dac.h"
|
||||
#endif
|
||||
@ -649,14 +645,6 @@ int esp32_bringup(void)
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifdef CONFIG_ESP32_RMT
|
||||
ret = board_rmt_initialize(RMT_CHANNEL, RMT_OUTPUT_PIN);
|
||||
if (ret < 0)
|
||||
{
|
||||
syslog(LOG_ERR, "ERROR: board_rmt_initialize() failed: %d\n", ret);
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifdef CONFIG_DAC
|
||||
ret = board_dac_initialize(CONFIG_ESP32_DAC_DEVPATH);
|
||||
if (ret < 0)
|
||||
|
Loading…
Reference in New Issue
Block a user