Cleanup lighting code and add ringbuffer to the LED tx UART
parent
38670e8014
commit
61bb07fa11
|
|
@ -15,12 +15,37 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "anne_pro.h"
|
#include "anne_pro.h"
|
||||||
|
#include "uart_tx_ringbuf.h"
|
||||||
#include "ch.h"
|
#include "ch.h"
|
||||||
#include "hal.h"
|
#include "hal.h"
|
||||||
|
|
||||||
static UARTDriver *LED_UART = &UARTD3;
|
#define LED_UART (&UARTD3)
|
||||||
|
/* UART transmission ringbuffer for the LED UART */
|
||||||
|
static uint8_t led_uart_tx_buffer[256] = {0};
|
||||||
|
static uart_tx_ringbuf_t led_uart_ringbuf = {
|
||||||
|
.buf = led_uart_tx_buffer,
|
||||||
|
.uart = LED_UART,
|
||||||
|
.size = sizeof(led_uart_tx_buffer),
|
||||||
|
.sending_elements = 0,
|
||||||
|
.head = 0,
|
||||||
|
.tail = 0,
|
||||||
|
};
|
||||||
|
|
||||||
|
/* Should the main loop try a transmission, used when transmission finishes
|
||||||
|
to send any data that might have been added to the buffer while the
|
||||||
|
transmission was in progress.
|
||||||
|
*/
|
||||||
|
static volatile bool led_uart_try_transmission = false;
|
||||||
|
|
||||||
|
/* Handler for finsihed LED UART transmissions */
|
||||||
|
static void led_uart_txend(UARTDriver *uart) {
|
||||||
|
uart_tx_ringbuf_finish_transmission(&led_uart_ringbuf);
|
||||||
|
led_uart_try_transmission = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* LED UART configuration */
|
||||||
static UARTConfig led_uart_cfg = {
|
static UARTConfig led_uart_cfg = {
|
||||||
.txend1_cb = NULL,
|
.txend1_cb = led_uart_txend,
|
||||||
.txend2_cb = NULL,
|
.txend2_cb = NULL,
|
||||||
.rxend_cb = NULL,
|
.rxend_cb = NULL,
|
||||||
.rxchar_cb = NULL,
|
.rxchar_cb = NULL,
|
||||||
|
|
@ -32,13 +57,13 @@ static UARTConfig led_uart_cfg = {
|
||||||
};
|
};
|
||||||
|
|
||||||
/* State of the leds on the keyboard */
|
/* State of the leds on the keyboard */
|
||||||
volatile bool leds_enabled = false;
|
static volatile bool leds_enabled = false;
|
||||||
|
|
||||||
/* Buffer for the keystate packet */
|
/* Buffer for the keystate packet */
|
||||||
static uint8_t keystate[12] = {9, 10, 7, 0};
|
static uint8_t keystate[12] = {9, 10, 7, 0};
|
||||||
|
|
||||||
/* Process the Anne Pro custom keycodes */
|
/* Update the dynamic lighting packet based on a keypress */
|
||||||
bool process_record_kb(uint16_t keycode, keyrecord_t *record) {
|
void update_dynamic_lighting(keyrecord_t *record) {
|
||||||
/* Only update dynamic lighting modes when leds are enabled */
|
/* Only update dynamic lighting modes when leds are enabled */
|
||||||
if (leds_enabled) {
|
if (leds_enabled) {
|
||||||
/* Calculate the position of the key that was pressed */
|
/* Calculate the position of the key that was pressed */
|
||||||
|
|
@ -56,8 +81,13 @@ bool process_record_kb(uint16_t keycode, keyrecord_t *record) {
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Send the keystate to the LED controller */
|
/* Send the keystate to the LED controller */
|
||||||
uartStartSend(LED_UART, 12, keystate);
|
uart_tx_ringbuf_write(&led_uart_ringbuf, 12, keystate);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Process the Anne Pro custom keycodes */
|
||||||
|
bool process_record_kb(uint16_t keycode, keyrecord_t *record) {
|
||||||
|
update_dynamic_lighting(record);
|
||||||
|
|
||||||
switch (keycode) {
|
switch (keycode) {
|
||||||
case APL_RGB:
|
case APL_RGB:
|
||||||
|
|
@ -65,28 +95,28 @@ bool process_record_kb(uint16_t keycode, keyrecord_t *record) {
|
||||||
if (record->event.pressed) {
|
if (record->event.pressed) {
|
||||||
leds_enabled = !leds_enabled;
|
leds_enabled = !leds_enabled;
|
||||||
if (leds_enabled) {
|
if (leds_enabled) {
|
||||||
uartStartSend(LED_UART, 3, "\x09\x01\x01");
|
uart_tx_ringbuf_write(&led_uart_ringbuf, 3, "\x09\x01\x01");
|
||||||
} else {
|
} else {
|
||||||
uartStartSend(LED_UART, 4, "\x09\x02\x01\x00");
|
uart_tx_ringbuf_write(&led_uart_ringbuf, 4, "\x09\x02\x01\x00");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
case APL_RAT:
|
case APL_RAT:
|
||||||
/* Change the animation rate */
|
/* Change the animation rate */
|
||||||
if (leds_enabled && record->event.pressed) {
|
if (leds_enabled && record->event.pressed) {
|
||||||
uartStartSend(LED_UART, 6, "\x09\x04\x05\x00\x01\x00");
|
uart_tx_ringbuf_write(&led_uart_ringbuf, 6, "\x09\x04\x05\x00\x01\x00");
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
case APL_BRT:
|
case APL_BRT:
|
||||||
/* Change the brightness */
|
/* Change the brightness */
|
||||||
if (leds_enabled && record->event.pressed) {
|
if (leds_enabled && record->event.pressed) {
|
||||||
uartStartSend(LED_UART, 6, "\x09\x04\x05\x00\x00\x01");
|
uart_tx_ringbuf_write(&led_uart_ringbuf, 6, "\x09\x04\x05\x00\x00\x01");
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
case APL_MOD:
|
case APL_MOD:
|
||||||
/* Change the lighting mode */
|
/* Change the lighting mode */
|
||||||
if (leds_enabled && record->event.pressed) {
|
if (leds_enabled && record->event.pressed) {
|
||||||
uartStartSend(LED_UART, 6, "\x09\x04\x05\x01\x00\x00");
|
uart_tx_ringbuf_write(&led_uart_ringbuf, 6, "\x09\x04\x05\x01\x00\x00");
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
default:
|
default:
|
||||||
|
|
@ -96,7 +126,7 @@ bool process_record_kb(uint16_t keycode, keyrecord_t *record) {
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Initialize custom keyboard features */
|
/* Initialize custom keyboard features */
|
||||||
void keyboard_post_init_kb(void) {
|
void keyboard_pre_init_kb(void) {
|
||||||
/* Turn on lighting controller */
|
/* Turn on lighting controller */
|
||||||
setPinOutput(C15);
|
setPinOutput(C15);
|
||||||
writePinLow(C15);
|
writePinLow(C15);
|
||||||
|
|
@ -109,9 +139,25 @@ void keyboard_post_init_kb(void) {
|
||||||
palSetPadMode(GPIOB, 10, PAL_MODE_ALTERNATE(7));
|
palSetPadMode(GPIOB, 10, PAL_MODE_ALTERNATE(7));
|
||||||
palSetPadMode(GPIOB, 11, PAL_MODE_ALTERNATE(7));
|
palSetPadMode(GPIOB, 11, PAL_MODE_ALTERNATE(7));
|
||||||
|
|
||||||
|
keyboard_pre_init_user();
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Turn on the lighting when init is finished */
|
||||||
|
void keyboard_post_init_kb(void) {
|
||||||
/* Send 'set theme' command to lighting controller */
|
/* Send 'set theme' command to lighting controller */
|
||||||
leds_enabled = true;
|
leds_enabled = true;
|
||||||
uartStartSend(LED_UART, 4, "\x09\x02\x01\x01");
|
uart_tx_ringbuf_write(&led_uart_ringbuf, 3, "\x09\x01\x01");
|
||||||
|
|
||||||
matrix_init_user();
|
keyboard_post_init_user();
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Start transmissions when the flag is set */
|
||||||
|
void matrix_scan_kb(void) {
|
||||||
|
if (led_uart_try_transmission) {
|
||||||
|
uart_tx_ringbuf_start_transmission(&led_uart_ringbuf);
|
||||||
|
led_uart_try_transmission = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Run matrix_scan_user code */
|
||||||
|
matrix_scan_user();
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,102 @@
|
||||||
|
/* Copyright 2019 Michiel Visser
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 2 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
#include "hal.h"
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
uint8_t *buf;
|
||||||
|
UARTDriver *uart;
|
||||||
|
size_t size;
|
||||||
|
volatile size_t sending_elements;
|
||||||
|
volatile size_t head;
|
||||||
|
volatile size_t tail;
|
||||||
|
} uart_tx_ringbuf_t;
|
||||||
|
|
||||||
|
int uart_tx_ringbuf_elements(uart_tx_ringbuf_t *ringbuf) {
|
||||||
|
return ringbuf->head - ringbuf->tail;
|
||||||
|
}
|
||||||
|
|
||||||
|
int uart_tx_ringbuf_space(uart_tx_ringbuf_t *ringbuf) {
|
||||||
|
return ringbuf->size - uart_tx_ringbuf_elements(ringbuf);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool uart_tx_ringbuf_empty(uart_tx_ringbuf_t *ringbuf) {
|
||||||
|
return uart_tx_ringbuf_elements(ringbuf) == 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool uart_tx_ringbuf_full(uart_tx_ringbuf_t *ringbuf) {
|
||||||
|
return uart_tx_ringbuf_elements(ringbuf) == ringbuf->size;
|
||||||
|
}
|
||||||
|
|
||||||
|
void uart_tx_ringbuf_start_transmission(uart_tx_ringbuf_t *ringbuf) {
|
||||||
|
/* Do not start transmission when one is already in progress */
|
||||||
|
if (ringbuf->sending_elements != 0) return;
|
||||||
|
|
||||||
|
size_t elements = uart_tx_ringbuf_elements(ringbuf);
|
||||||
|
/* Do not start transmission when ringbuffer is empty */
|
||||||
|
if (elements == 0) return;
|
||||||
|
|
||||||
|
/* Get the offset of the first and last item */
|
||||||
|
size_t first_offset = (ringbuf->tail & (ringbuf->size - 1));
|
||||||
|
|
||||||
|
/* Determine if this wraps around the end of the buffer */
|
||||||
|
if (first_offset + elements >= ringbuf->size) {
|
||||||
|
/* If so only send until the end of the buffer */
|
||||||
|
elements = ringbuf->size - first_offset;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Send the selected elements */
|
||||||
|
ringbuf->sending_elements = elements;
|
||||||
|
uartStartSend(ringbuf->uart, elements, ringbuf->buf + first_offset);
|
||||||
|
}
|
||||||
|
|
||||||
|
void uart_tx_ringbuf_finish_transmission(uart_tx_ringbuf_t *ringbuf) {
|
||||||
|
/* Update the tail of the ringbuffer */
|
||||||
|
ringbuf->tail += ringbuf->sending_elements;
|
||||||
|
/* Clear the sending elements */
|
||||||
|
ringbuf->sending_elements = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool uart_tx_ringbuf_put(uart_tx_ringbuf_t *ringbuf, uint8_t c) {
|
||||||
|
/* If the ringbuffer is full, ignore the request */
|
||||||
|
if (uart_tx_ringbuf_full(ringbuf)) return false;
|
||||||
|
|
||||||
|
/* Put the character into the ringbuffer */
|
||||||
|
size_t head_offset = (ringbuf->head & (ringbuf->size - 1));
|
||||||
|
ringbuf->buf[head_offset] = c;
|
||||||
|
ringbuf->head++;
|
||||||
|
|
||||||
|
/* Try to start the transmission immediately */
|
||||||
|
uart_tx_ringbuf_start_transmission(ringbuf);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool uart_tx_ringbuf_write(uart_tx_ringbuf_t *ringbuf, size_t count, void *data) {
|
||||||
|
/* If there is not enough space, ignore the request */
|
||||||
|
if (uart_tx_ringbuf_space(ringbuf) < count) return false;
|
||||||
|
|
||||||
|
for (int i = 0; i < count; i++) {
|
||||||
|
/* Put the character into the ringbuffer */
|
||||||
|
size_t head_offset = (ringbuf->head & (ringbuf->size - 1));
|
||||||
|
ringbuf->buf[head_offset] = ((uint8_t *) data)[i];
|
||||||
|
ringbuf->head++;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Try to start the transmission immediately */
|
||||||
|
uart_tx_ringbuf_start_transmission(ringbuf);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
Loading…
Reference in New Issue