mirror of https://github.com/midilab/uClock
commit
e6dc1fd7aa
@ -0,0 +1,89 @@ |
|||||||
|
/*
|
||||||
|
* USB/Uart MIDI Sync Box
|
||||||
|
*
|
||||||
|
* This example code is in the public domain. |
||||||
|
*
|
||||||
|
*/ |
||||||
|
|
||||||
|
#include <Adafruit_TinyUSB.h> |
||||||
|
#include <MIDI.h> |
||||||
|
|
||||||
|
#include <uClock.h> |
||||||
|
|
||||||
|
// Instantiate the MIDI interfaces
|
||||||
|
Adafruit_USBD_MIDI usb_midi; |
||||||
|
MIDI_CREATE_INSTANCE(Adafruit_USBD_MIDI, usb_midi, MIDI_USB); |
||||||
|
MIDI_CREATE_INSTANCE(HardwareSerial, Serial1, MIDI); |
||||||
|
|
||||||
|
// Do your rpi 2040 has a ws2812 RGB LED? set the pin!
|
||||||
|
// otherwise keep it commented for normal LED_BUILTIN led blinking
|
||||||
|
#define WS2812_BUILTIN_LED 16 |
||||||
|
|
||||||
|
uint8_t bpm_blink_timer = 1; |
||||||
|
void handle_bpm_led(uint32_t tick) |
||||||
|
{ |
||||||
|
// BPM led indicator
|
||||||
|
if ( !(tick % (96)) || (tick == 1) ) { // first of 4 quarter pulse will flash longer
|
||||||
|
bpm_blink_timer = 8; |
||||||
|
ledOn(); |
||||||
|
} else if ( !(tick % (24)) ) { // each quarter led on
|
||||||
|
bpm_blink_timer = 1; |
||||||
|
ledOn(); |
||||||
|
} else if ( !(tick % bpm_blink_timer) ) { // get led off
|
||||||
|
ledOff(); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
// Internal clock handlers
|
||||||
|
void onSync24Callback(uint32_t tick) { |
||||||
|
// Send MIDI_CLOCK to external gears
|
||||||
|
MIDI.sendRealTime(midi::Clock); |
||||||
|
MIDI_USB.sendRealTime(midi::Clock); |
||||||
|
// blink tempo
|
||||||
|
handle_bpm_led(tick); |
||||||
|
} |
||||||
|
|
||||||
|
void onClockStart() { |
||||||
|
MIDI.sendRealTime(midi::Start); |
||||||
|
MIDI_USB.sendRealTime(midi::Start); |
||||||
|
} |
||||||
|
|
||||||
|
void onClockStop() { |
||||||
|
MIDI.sendRealTime(midi::Stop); |
||||||
|
MIDI_USB.sendRealTime(midi::Stop); |
||||||
|
} |
||||||
|
|
||||||
|
void setup() { |
||||||
|
#if defined(ARDUINO_ARCH_MBED) && defined(ARDUINO_ARCH_RP2040) |
||||||
|
// Manual begin() is required on core without built-in support for TinyUSB such as mbed rp2040
|
||||||
|
TinyUSB_Device_Init(0); |
||||||
|
#endif |
||||||
|
|
||||||
|
// Initialize USB midi stack
|
||||||
|
MIDI_USB.begin(MIDI_CHANNEL_OMNI); |
||||||
|
// Initialize UART midi stack
|
||||||
|
MIDI.begin(MIDI_CHANNEL_OMNI); |
||||||
|
|
||||||
|
// Initialize builtin led for clock timer blinking
|
||||||
|
initBlinkLed(); |
||||||
|
|
||||||
|
// Setup our clock system
|
||||||
|
// Inits the clock
|
||||||
|
uClock.init(); |
||||||
|
// Set the callback function for the clock output to send MIDI Sync message.
|
||||||
|
uClock.setOnSync24(onSync24Callback); |
||||||
|
// Set the callback function for MIDI Start and Stop messages.
|
||||||
|
uClock.setOnClockStart(onClockStart);
|
||||||
|
uClock.setOnClockStop(onClockStop); |
||||||
|
// Set the clock BPM to 126 BPM
|
||||||
|
uClock.setTempo(126); |
||||||
|
// Starts the clock, tick-tac-tick-tac..
|
||||||
|
uClock.start(); |
||||||
|
} |
||||||
|
|
||||||
|
// Do it whatever to interface with Clock.stop(), Clock.start(), Clock.setTempo() and integrate your environment...
|
||||||
|
void loop() { |
||||||
|
// handle midi input?
|
||||||
|
MIDI.read(); |
||||||
|
MIDI_USB.read(); |
||||||
|
} |
@ -0,0 +1,37 @@ |
|||||||
|
#if defined(WS2812_BUILTIN_LED) |
||||||
|
#include <Adafruit_NeoPixel.h> |
||||||
|
#define NUMPIXELS 1 |
||||||
|
Adafruit_NeoPixel pixels(NUMPIXELS, WS2812_BUILTIN_LED, NEO_GRB + NEO_KHZ800); |
||||||
|
#endif |
||||||
|
|
||||||
|
// check the pinage for BUILTIN LED of your model in case LED_BUILTIN wont ligth up
|
||||||
|
// this is valid only if you're not using rgb version ws2812 (WS2812_BUILTIN_LED)
|
||||||
|
//#define LED_BUILTIN PIN_LED_B
|
||||||
|
|
||||||
|
void initBlinkLed() { |
||||||
|
#if defined(WS2812_BUILTIN_LED) |
||||||
|
// use adafruit neo pixel
|
||||||
|
pixels.begin(); |
||||||
|
#else |
||||||
|
// normal led pin
|
||||||
|
pinMode(LED_BUILTIN, OUTPUT); |
||||||
|
#endif |
||||||
|
} |
||||||
|
|
||||||
|
void ledOn() { |
||||||
|
#if defined(WS2812_BUILTIN_LED) |
||||||
|
pixels.setPixelColor(0, pixels.Color(0, 0, 20)); |
||||||
|
pixels.show(); // turn the LED on (HIGH is the voltage level)
|
||||||
|
#else |
||||||
|
digitalWrite(LED_BUILTIN, LOW); |
||||||
|
#endif |
||||||
|
} |
||||||
|
|
||||||
|
void ledOff() { |
||||||
|
#if defined(WS2812_BUILTIN_LED) |
||||||
|
pixels.setPixelColor(0, pixels.Color(0, 0, 0)); |
||||||
|
pixels.show(); |
||||||
|
#else |
||||||
|
digitalWrite(LED_BUILTIN, HIGH); |
||||||
|
#endif |
||||||
|
} |
@ -1,10 +1,10 @@ |
|||||||
name=uClock |
name=uClock |
||||||
version=2.0.0 |
version=2.1.0 |
||||||
author=Romulo Silva <contact@midilab.co> |
author=Romulo Silva <contact@midilab.co> |
||||||
maintainer=Romulo Silva <contact@midilab.co> |
maintainer=Romulo Silva <contact@midilab.co> |
||||||
sentence=BPM clock generator for Arduino platform. |
sentence=BPM clock generator for Arduino platform. |
||||||
paragraph=A Library to implement BPM clock tick calls using hardware interruption. Supported and tested on AVR boards(ATmega168/328, ATmega16u4/32u4 and ATmega2560) and ARM boards(Teensy, Seedstudio XIAO M0. ESP32 and STM32) |
paragraph=A Library to implement BPM clock tick calls using hardware interruption. Supported and tested on AVR boards(ATmega168/328, ATmega16u4/32u4 and ATmega2560) and ARM boards(RPI2040, Teensy, Seedstudio XIAO M0 and ESP32) |
||||||
category=Timing |
category=Timing |
||||||
url=https://github.com/midilab/uClock |
url=https://github.com/midilab/uClock |
||||||
architectures=avr,arm,samd,stm32,esp32 |
architectures=avr,arm,samd,stm32,esp32,rp2040 |
||||||
includes=uClock.h |
includes=uClock.h |
||||||
|
@ -0,0 +1,30 @@ |
|||||||
|
#include <Arduino.h> |
||||||
|
#include "pico/sync.h" |
||||||
|
|
||||||
|
// RPi-specific timer
|
||||||
|
struct repeating_timer timer; |
||||||
|
|
||||||
|
#define ATOMIC(X) { uint32_t __interrupt_mask = save_and_disable_interrupts(); X; restore_interrupts(__interrupt_mask); } |
||||||
|
|
||||||
|
// forward declaration of uClockHandler
|
||||||
|
void uClockHandler(); |
||||||
|
|
||||||
|
// ISR handler -- called when tick happens
|
||||||
|
bool handlerISR(repeating_timer *timer) |
||||||
|
{ |
||||||
|
uClockHandler(); |
||||||
|
|
||||||
|
return true; |
||||||
|
} |
||||||
|
|
||||||
|
void initTimer(uint32_t init_clock) { |
||||||
|
// set up RPi interrupt timer
|
||||||
|
// todo: actually should be -init_clock so that timer is set to start init_clock us after last tick, instead of init_clock us after finished processing last tick!
|
||||||
|
add_repeating_timer_us(init_clock, &handlerISR, NULL, &timer); |
||||||
|
} |
||||||
|
|
||||||
|
void setTimer(uint32_t us_interval) { |
||||||
|
cancel_repeating_timer(&timer); |
||||||
|
// todo: actually should be -us_interval so that timer is set to start init_clock us after last tick, instead of init_clock us after finished processing last tick!
|
||||||
|
add_repeating_timer_us(us_interval, &handlerISR, NULL, &timer); |
||||||
|
} |
Loading…
Reference in new issue