Skip to content

Commit c5b7570

Browse files
committed
Merge remote-tracking branch 'bcmi/add_renesas_port'
2 parents 9fe7546 + 11a5c8c commit c5b7570

File tree

4 files changed

+222
-1
lines changed

4 files changed

+222
-1
lines changed

library.properties

+1-1
Original file line numberDiff line numberDiff line change
@@ -6,4 +6,4 @@ sentence=Allows Arduino boards to control a variety of servo motors.
66
paragraph=This library can control a great number of servos.<br />It makes careful use of timers: the library can control 12 servos using only 1 timer.<br />On the Arduino Due you can control up to 60 servos.
77
category=Device Control
88
url=https://www.arduino.cc/reference/en/libraries/servo/
9-
architectures=avr,megaavr,sam,samd,nrf52,stm32f4,mbed,mbed_nano,mbed_portenta,mbed_rp2040
9+
architectures=avr,megaavr,sam,samd,nrf52,stm32f4,mbed,mbed_nano,mbed_portenta,mbed_rp2040,renesas

src/Servo.h

+2
Original file line numberDiff line numberDiff line change
@@ -73,6 +73,8 @@
7373
#include "megaavr/ServoTimers.h"
7474
#elif defined(ARDUINO_ARCH_MBED)
7575
#include "mbed/ServoTimers.h"
76+
#elif defined(ARDUINO_ARCH_RENESAS)
77+
#include "renesas/ServoTimers.h"
7678
#else
7779
#error "This library only supports boards with an AVR, SAM, SAMD, NRF52 or STM32F4 processor."
7880
#endif

src/renesas/Servo.cpp

+218
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,218 @@
1+
/* The MIT License (MIT)
2+
*
3+
* Copyright (c) 2022 Arduino SA
4+
*
5+
* Permission is hereby granted, free of charge, to any person obtaining a copy
6+
* of this software and associated documentation files (the "Software"), to deal
7+
* in the Software without restriction, including without limitation the rights
8+
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9+
* copies of the Software, and to permit persons to whom the Software is
10+
* furnished to do so, subject to the following conditions:
11+
*
12+
* The above copyright notice and this permission notice shall be included in
13+
* all copies or substantial portions of the Software.
14+
*
15+
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16+
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17+
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18+
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19+
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20+
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
21+
* THE SOFTWARE.
22+
*/
23+
#if defined(ARDUINO_ARCH_RENESAS)
24+
25+
#include "Arduino.h"
26+
#include "Servo.h"
27+
#include "ServoTimers.h"
28+
#include "math.h"
29+
#include "FspTimer.h"
30+
31+
#define SERVO_MAX_SERVOS (_Nbr_16timers * SERVOS_PER_TIMER)
32+
#define SERVO_INVALID_INDEX (255)
33+
// Lower the timer ticks for finer resolution.
34+
#define SERVO_TIMER_TICK_US (100)
35+
#define SERVO_US_PER_CYCLE (20000)
36+
#define SERVO_IO_PORT_ADDR(pn) &((R_PORT0 + ((uint32_t) (R_PORT1 - R_PORT0) * (pn)))->PCNTR3)
37+
38+
// Internal Servo sturct to keep track of RA configuration.
39+
typedef struct {
40+
// Servo period in microseconds.
41+
uint32_t period_us;
42+
// Store min/max pulse width here, because min/max in
43+
// Servo class are not wide enough for the pulse width.
44+
uint32_t period_min;
45+
uint32_t period_max;
46+
// Period period_count in microseconds.
47+
uint32_t period_count;
48+
// Internal FSP GPIO port/pin control bits.
49+
volatile uint32_t *io_port;
50+
uint32_t io_mask;
51+
} ra_servo_t;
52+
53+
// Keep track of the total number of servos attached.
54+
static size_t n_servos=0;
55+
static ra_servo_t ra_servos[SERVO_MAX_SERVOS];
56+
57+
static FspTimer servo_timer;
58+
static bool servo_timer_started = false;
59+
void servo_timer_callback(timer_callback_args_t *args);
60+
61+
static int servo_timer_config(uint32_t period_us)
62+
{
63+
static bool configured = false;
64+
if (configured == false) {
65+
// Configure and enable the servo timer.
66+
uint8_t type = 0;
67+
int8_t channel = FspTimer::get_available_timer(type);
68+
if (channel != -1) {
69+
servo_timer.begin(TIMER_MODE_PERIODIC, type, channel,
70+
1000000.0f/period_us, 50.0f, servo_timer_callback, nullptr);
71+
servo_timer.setup_overflow_irq();
72+
servo_timer.open();
73+
servo_timer.stop();
74+
configured = true;
75+
}
76+
}
77+
return configured ? 0 : -1;
78+
}
79+
80+
static int servo_timer_start()
81+
{
82+
// Start the timer if it's not started
83+
if (servo_timer_started == false &&
84+
servo_timer.start() == false) {
85+
return -1;
86+
}
87+
servo_timer_started = true;
88+
return 0;
89+
}
90+
91+
static int servo_timer_stop()
92+
{
93+
// Start the timer if it's not started
94+
if (servo_timer_started == true &&
95+
servo_timer.stop() == false) {
96+
return -1;
97+
}
98+
servo_timer_started = false;
99+
return 0;
100+
}
101+
102+
void servo_timer_callback(timer_callback_args_t *args)
103+
{
104+
for (size_t i=0; i<SERVO_MAX_SERVOS; i++) {
105+
ra_servo_t *servo = &ra_servos[i];
106+
if (servo->period_us) {
107+
servo->period_count += SERVO_TIMER_TICK_US;
108+
if (servo->period_count <= servo->period_us) {
109+
*servo->io_port = (uint32_t) servo->io_mask;
110+
} else {
111+
*servo->io_port = (uint32_t) (servo->io_mask << 16);
112+
}
113+
if (servo->period_count == SERVO_US_PER_CYCLE) {
114+
servo->period_count = 0;
115+
}
116+
}
117+
}
118+
}
119+
120+
Servo::Servo()
121+
{
122+
servoIndex = SERVO_INVALID_INDEX;
123+
}
124+
125+
uint8_t Servo::attach(int pin)
126+
{
127+
return attach(pin, MIN_PULSE_WIDTH, MAX_PULSE_WIDTH);
128+
}
129+
130+
uint8_t Servo::attach(int pin, int min, int max)
131+
{
132+
//assert(pin < NUM_DIGITAL_PINS); ?
133+
if (n_servos == SERVO_MAX_SERVOS) {
134+
return 0;
135+
}
136+
137+
// Try to find a free servo slot.
138+
ra_servo_t *servo = NULL;
139+
bsp_io_port_pin_t io_pin = g_pin_cfg[pin].pin;
140+
for (size_t i=0; i<SERVO_MAX_SERVOS; i++) {
141+
servo = &ra_servos[i];
142+
if (servo->period_us == 0) {
143+
n_servos++;
144+
servoIndex = i;
145+
servo->period_min = min;
146+
servo->period_max = max;
147+
servo->io_mask = (1U << (io_pin & 0xFF));
148+
servo->io_port = SERVO_IO_PORT_ADDR(((io_pin >> 8U) & 0xFF));
149+
writeMicroseconds(DEFAULT_PULSE_WIDTH);
150+
break;
151+
}
152+
}
153+
154+
if (servoIndex == SERVO_INVALID_INDEX) {
155+
return 0;
156+
}
157+
158+
// Configure GPIO pin for the servo.
159+
R_IOPORT_PinCfg(&g_ioport_ctrl, io_pin,
160+
IOPORT_CFG_PORT_DIRECTION_OUTPUT | IOPORT_CFG_PORT_OUTPUT_HIGH);
161+
162+
// Configure and start the timer if it's not started.
163+
if (servo_timer_config(SERVO_TIMER_TICK_US) != 0 ||
164+
servo_timer_start() != 0) {
165+
return 0;
166+
}
167+
return 1;
168+
}
169+
170+
void Servo::detach()
171+
{
172+
if (servoIndex != SERVO_INVALID_INDEX) {
173+
ra_servo_t *servo = &ra_servos[servoIndex];
174+
servo_timer_stop();
175+
servo->period_us = 0;
176+
if (--n_servos) {
177+
servo_timer_start();
178+
}
179+
servoIndex = SERVO_INVALID_INDEX;
180+
}
181+
}
182+
183+
void Servo::write(int angle)
184+
{
185+
if (servoIndex != SERVO_INVALID_INDEX) {
186+
ra_servo_t *servo = &ra_servos[servoIndex];
187+
angle = constrain(angle, 0, 180);
188+
writeMicroseconds(map(angle, 0, 180, servo->period_min, servo->period_max));
189+
}
190+
}
191+
192+
int Servo::read()
193+
{
194+
if (servoIndex != SERVO_INVALID_INDEX) {
195+
ra_servo_t *servo = &ra_servos[servoIndex];
196+
return map(servo->period_us, servo->period_min, servo->period_max, 0, 180);
197+
}
198+
return 0;
199+
}
200+
201+
void Servo::writeMicroseconds(int us)
202+
{
203+
if (servoIndex != SERVO_INVALID_INDEX) {
204+
ra_servo_t *servo = &ra_servos[servoIndex];
205+
servo->period_count = 0;
206+
servo->period_us = constrain(us, servo->period_min, servo->period_max);
207+
}
208+
}
209+
210+
int Servo::readMicroseconds()
211+
{
212+
if (servoIndex != SERVO_INVALID_INDEX) {
213+
ra_servo_t *servo = &ra_servos[servoIndex];
214+
return servo->period_us;
215+
}
216+
return 0;
217+
}
218+
#endif // defined(ARDUINO_ARCH_RENESAS)

src/renesas/ServoTimers.h

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
#define _Nbr_16timers 1

0 commit comments

Comments
 (0)