Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Example for controlling a chain of WS2812 LEDs using ESP32 RMT periph… #1849

Merged
merged 1 commit into from
Jul 30, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
23 changes: 23 additions & 0 deletions examples/ws2812esp32rmt/Kconfig
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
#
# For a description of the syntax of this configuration file,
# see the file kconfig-language.txt in the NuttX tools repository.
#

config EXAMPLES_WS2812_ESP32_RMT
tristate "ws2812 Demo Program using ESP32_RMT"
depends on ESP32_RMT
default n
---help---
Enable the ws2812 demo, using ESP32 RMT peripheral to generate the pulses

if EXAMPLES_WS2812_ESP32_RMT

config EXAMPLES_WS2812_ESP32_RMT_PRIORITY
int "Task Priority"
default 100

config EXAMPLES_WS2812_ESP32_RMT_STACKSIZE
int "Stack Size"
default DEFAULT_TASK_STACKSIZE

endif
23 changes: 23 additions & 0 deletions examples/ws2812esp32rmt/Make.defs
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
############################################################################
# apps/examples/ws2812/Make.defs
#
# 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.
#
############################################################################

ifneq ($(CONFIG_EXAMPLES_WS2812_ESP32_RMT),)
CONFIGURED_APPS += $(APPDIR)/examples/ws2812esp32rmt
endif
32 changes: 32 additions & 0 deletions examples/ws2812esp32rmt/Makefile
Original file line number Diff line number Diff line change
@@ -0,0 +1,32 @@
############################################################################
# apps/examples/ws2812/Makefile
#
# 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.
#
############################################################################

include $(APPDIR)/Make.defs

# MTD R/W buffer test Example

PROGNAME = ws2812esp32rmt
PRIORITY = $(CONFIG_EXAMPLES_WS2812_ESP32_RMT_PRIORITY)
STACKSIZE = $(CONFIG_EXAMPLES_WS2812_ESP32_RMT_STACKSIZE)
MODULE = $(CONFIG_EXAMPLES_WS2812_ESP32_RMT)

MAINSRC = ws2812esp32rmt_main.c

include $(APPDIR)/Application.mk
279 changes: 279 additions & 0 deletions examples/ws2812esp32rmt/ws2812esp32rmt_main.c
Original file line number Diff line number Diff line change
@@ -0,0 +1,279 @@
/****************************************************************************
* apps/examples/ws2812esp32rmt/ws2812esp32rmt_main.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 <errno.h>
#include <fcntl.h>
#include <inttypes.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <sys/ioctl.h>
#include <unistd.h>

/****************************************************************************
* 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
****************************************************************************/

struct led_s
{
uint8_t r;
uint8_t g;
uint8_t b;
};

/****************************************************************************
* Private Data
****************************************************************************/

static int fd;
static uint32_t *rmt_buffer;
static uint32_t rmt_buffer_len_in_words;
static uint32_t leds_buffer_len_in_bytes;
static struct led_s *leds_buffer;

/****************************************************************************
* Private Functions
****************************************************************************/

static uint32_t map_byte_to_words(uint8_t byte, uint32_t *dst)
{
uint32_t ret = 0;
uint8_t mask = 0x80;
for (int i = 0; i < 8; i++)
{
uint8_t bit = byte&mask;
mask >>= 1;
uint32_t word;
if (bit)
{
word = (T1L << 16) | (0x8000 | T1H);
}
else
{
word = (T0L << 16) | (0x8000 | T0H);
}

*dst = word;
dst++;
ret++;
}

return ret;
}

static int map_leds_to_words(struct led_s *leds,
uint32_t n_leds,
uint32_t *dst,
uint32_t dst_len_in_words
)
{
int ret = OK;

if (!dst || !leds)
{
return -1;
}

uint32_t required_words = n_leds * 3 * 8;

if (required_words >= dst_len_in_words)
{
_err("Required %d words, but buffer has only:%d",
required_words, dst_len_in_words
);
return -1;
}

uint32_t dst_offset = 0;
for (uint32_t led_idx = 0; led_idx < n_leds; led_idx++)
{
dst_offset += map_byte_to_words(leds[led_idx].g, dst + dst_offset);
dst_offset += map_byte_to_words(leds[led_idx].r, dst + dst_offset);
dst_offset += map_byte_to_words(leds[led_idx].b, dst + dst_offset);
}

return ret;
}

/****************************************************************************
* Public Functions
****************************************************************************/

/****************************************************************************
* ws2812esp32rmt_main
****************************************************************************/

int main(int argc, FAR char *argv[])
{
if (argc < 2)
{
printf("usage: %s channel led_count\n", argv[0]);
exit(1);
}

int ch_idx = atoi(argv[1]);
if (ch_idx >= 8)
{
printf("channel must be int the range 0~7\n");
exit(1);
}

uint32_t n_leds = atoi(argv[2]);

char dev_name[50];
snprintf(dev_name, 50, "/dev/rmt%d", ch_idx);
fd = open(dev_name, O_WRONLY);
if (fd < 0)
{
fprintf(stderr,
"ws2812esp32rmt: open %s failed: %d\n",
dev_name,
errno);
goto errout_with_dev;
}

_info("size of (struct led_s) is:%d", sizeof(struct led_s));

leds_buffer_len_in_bytes = sizeof(struct led_s)*n_leds;
leds_buffer = (struct led_s *)malloc(leds_buffer_len_in_bytes);
_info("leds:%d - leds_buffer %p len:%d\n",
n_leds,
leds_buffer,
leds_buffer_len_in_bytes
);

if (!leds_buffer)
{
_err("failure allocating %d bytes for LED buffer",
leds_buffer_len_in_bytes
);
goto errout;
}

rmt_buffer_len_in_words = (leds_buffer_len_in_bytes * 8 + 1);
rmt_buffer = (uint32_t *)malloc(rmt_buffer_len_in_words * 4);
_info("words:%d - rmt_buffer %p rmt_buffer_len_in_words:%d\n",
rmt_buffer_len_in_words,
rmt_buffer,
rmt_buffer_len_in_words
);

if (!rmt_buffer)
{
_err("failure allocating %d words for RMT word buffer",
rmt_buffer_len_in_words
);
goto errout;
}

int pos = 0;
int direction = 0;

/* Run the display loop */

while (1)
{
int n_loops = 30;
for (int i = 0; i < n_loops; ++i)
{
for (int l = 0; l < n_leds; l++)
{
if (!direction)
leds_buffer[l].r = i;
else
leds_buffer[l].r = 30 - i;

leds_buffer[l].g = 0x00;
leds_buffer[l].b = 0x00;

if (l == pos)
{
leds_buffer[l].r = 0x00;
leds_buffer[l].g = 0xff;
leds_buffer[l].b = 0x00;
}
}

if (pos++ >= n_leds)
pos = 0;
int ret;
ret = map_leds_to_words(leds_buffer,
n_leds,
rmt_buffer,
rmt_buffer_len_in_words
);

if (ret)
{
_err("error mapping leds to words");
goto errout;
}

ret = write(fd, rmt_buffer, rmt_buffer_len_in_words * 4);

if (ret < 0)
{
_err("error writing to device, %d errno:%d", ret, errno);
goto errout_with_dev;
}

/****************************************************************************
* Must sleep so the WS2812 can understand the EOT
****************************************************************************/

usleep(50000);
}

direction = !direction;
}

free(leds_buffer);
free(rmt_buffer);
close(fd);
fflush(stdout);
return OK;

errout_with_dev:
close(fd);

errout:
fflush(stdout);
return ERROR;
}
Loading