New SPI API supporting DMA
[betaflight.git] / src / main / target / SITL / target.c
blob0b6b72f14caaeba5e4dd0dc4b651116aeee35b91
1 /*
2 * This file is part of Cleanflight and Betaflight.
4 * Cleanflight and Betaflight are free software. You can redistribute
5 * this software and/or modify this software under the terms of the
6 * GNU General Public License as published by the Free Software
7 * Foundation, either version 3 of the License, or (at your option)
8 * any later version.
10 * Cleanflight and Betaflight are distributed in the hope that they
11 * will be useful, but WITHOUT ANY WARRANTY; without even the implied
12 * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
13 * See the GNU General Public License for more details.
15 * You should have received a copy of the GNU General Public License
16 * along with this software.
18 * If not, see <http://www.gnu.org/licenses/>.
21 #include <stdint.h>
22 #include <stdio.h>
23 #include <stdlib.h>
24 #include <math.h>
25 #include <string.h>
27 #include <errno.h>
28 #include <time.h>
30 #include "common/maths.h"
32 #include "drivers/io.h"
33 #include "drivers/dma.h"
34 #include "drivers/motor.h"
35 #include "drivers/serial.h"
36 #include "drivers/serial_tcp.h"
37 #include "drivers/system.h"
38 #include "drivers/pwm_output.h"
39 #include "drivers/light_led.h"
41 #include "drivers/timer.h"
42 #include "drivers/timer_def.h"
43 const timerHardware_t timerHardware[1]; // unused
45 #include "drivers/accgyro/accgyro_fake.h"
46 #include "flight/imu.h"
48 #include "config/feature.h"
49 #include "config/config.h"
50 #include "scheduler/scheduler.h"
52 #include "pg/rx.h"
53 #include "pg/motor.h"
55 #include "rx/rx.h"
57 #include "dyad.h"
58 #include "target/SITL/udplink.h"
60 uint32_t SystemCoreClock;
62 static fdm_packet fdmPkt;
63 static servo_packet pwmPkt;
65 static struct timespec start_time;
66 static double simRate = 1.0;
67 static pthread_t tcpWorker, udpWorker;
68 static bool workerRunning = true;
69 static udpLink_t stateLink, pwmLink;
70 static pthread_mutex_t updateLock;
71 static pthread_mutex_t mainLoopLock;
73 int timeval_sub(struct timespec *result, struct timespec *x, struct timespec *y);
75 int lockMainPID(void) {
76 return pthread_mutex_trylock(&mainLoopLock);
79 #define RAD2DEG (180.0 / M_PI)
80 #define ACC_SCALE (256 / 9.80665)
81 #define GYRO_SCALE (16.4)
82 void sendMotorUpdate() {
83 udpSend(&pwmLink, &pwmPkt, sizeof(servo_packet));
85 void updateState(const fdm_packet* pkt) {
86 static double last_timestamp = 0; // in seconds
87 static uint64_t last_realtime = 0; // in uS
88 static struct timespec last_ts; // last packet
90 struct timespec now_ts;
91 clock_gettime(CLOCK_MONOTONIC, &now_ts);
93 const uint64_t realtime_now = micros64_real();
94 if (realtime_now > last_realtime + 500*1e3) { // 500ms timeout
95 last_timestamp = pkt->timestamp;
96 last_realtime = realtime_now;
97 sendMotorUpdate();
98 return;
101 const double deltaSim = pkt->timestamp - last_timestamp; // in seconds
102 if (deltaSim < 0) { // don't use old packet
103 return;
106 int16_t x,y,z;
107 x = constrain(-pkt->imu_linear_acceleration_xyz[0] * ACC_SCALE, -32767, 32767);
108 y = constrain(-pkt->imu_linear_acceleration_xyz[1] * ACC_SCALE, -32767, 32767);
109 z = constrain(-pkt->imu_linear_acceleration_xyz[2] * ACC_SCALE, -32767, 32767);
110 fakeAccSet(fakeAccDev, x, y, z);
111 // printf("[acc]%lf,%lf,%lf\n", pkt->imu_linear_acceleration_xyz[0], pkt->imu_linear_acceleration_xyz[1], pkt->imu_linear_acceleration_xyz[2]);
113 x = constrain(pkt->imu_angular_velocity_rpy[0] * GYRO_SCALE * RAD2DEG, -32767, 32767);
114 y = constrain(-pkt->imu_angular_velocity_rpy[1] * GYRO_SCALE * RAD2DEG, -32767, 32767);
115 z = constrain(-pkt->imu_angular_velocity_rpy[2] * GYRO_SCALE * RAD2DEG, -32767, 32767);
116 fakeGyroSet(fakeGyroDev, x, y, z);
117 // printf("[gyr]%lf,%lf,%lf\n", pkt->imu_angular_velocity_rpy[0], pkt->imu_angular_velocity_rpy[1], pkt->imu_angular_velocity_rpy[2]);
119 #if !defined(USE_IMU_CALC)
120 #if defined(SET_IMU_FROM_EULER)
121 // set from Euler
122 double qw = pkt->imu_orientation_quat[0];
123 double qx = pkt->imu_orientation_quat[1];
124 double qy = pkt->imu_orientation_quat[2];
125 double qz = pkt->imu_orientation_quat[3];
126 double ysqr = qy * qy;
127 double xf, yf, zf;
129 // roll (x-axis rotation)
130 double t0 = +2.0 * (qw * qx + qy * qz);
131 double t1 = +1.0 - 2.0 * (qx * qx + ysqr);
132 xf = atan2(t0, t1) * RAD2DEG;
134 // pitch (y-axis rotation)
135 double t2 = +2.0 * (qw * qy - qz * qx);
136 t2 = t2 > 1.0 ? 1.0 : t2;
137 t2 = t2 < -1.0 ? -1.0 : t2;
138 yf = asin(t2) * RAD2DEG; // from wiki
140 // yaw (z-axis rotation)
141 double t3 = +2.0 * (qw * qz + qx * qy);
142 double t4 = +1.0 - 2.0 * (ysqr + qz * qz);
143 zf = atan2(t3, t4) * RAD2DEG;
144 imuSetAttitudeRPY(xf, -yf, zf); // yes! pitch was inverted!!
145 #else
146 imuSetAttitudeQuat(pkt->imu_orientation_quat[0], pkt->imu_orientation_quat[1], pkt->imu_orientation_quat[2], pkt->imu_orientation_quat[3]);
147 #endif
148 #endif
150 #if defined(SIMULATOR_IMU_SYNC)
151 imuSetHasNewData(deltaSim*1e6);
152 imuUpdateAttitude(micros());
153 #endif
156 if (deltaSim < 0.02 && deltaSim > 0) { // simulator should run faster than 50Hz
157 // simRate = simRate * 0.5 + (1e6 * deltaSim / (realtime_now - last_realtime)) * 0.5;
158 struct timespec out_ts;
159 timeval_sub(&out_ts, &now_ts, &last_ts);
160 simRate = deltaSim / (out_ts.tv_sec + 1e-9*out_ts.tv_nsec);
162 // printf("simRate = %lf, millis64 = %lu, millis64_real = %lu, deltaSim = %lf\n", simRate, millis64(), millis64_real(), deltaSim*1e6);
164 last_timestamp = pkt->timestamp;
165 last_realtime = micros64_real();
167 last_ts.tv_sec = now_ts.tv_sec;
168 last_ts.tv_nsec = now_ts.tv_nsec;
170 pthread_mutex_unlock(&updateLock); // can send PWM output now
172 #if defined(SIMULATOR_GYROPID_SYNC)
173 pthread_mutex_unlock(&mainLoopLock); // can run main loop
174 #endif
177 static void* udpThread(void* data) {
178 UNUSED(data);
179 int n = 0;
181 while (workerRunning) {
182 n = udpRecv(&stateLink, &fdmPkt, sizeof(fdm_packet), 100);
183 if (n == sizeof(fdm_packet)) {
184 // printf("[data]new fdm %d\n", n);
185 updateState(&fdmPkt);
189 printf("udpThread end!!\n");
190 return NULL;
193 static void* tcpThread(void* data) {
194 UNUSED(data);
196 dyad_init();
197 dyad_setTickInterval(0.2f);
198 dyad_setUpdateTimeout(0.5f);
200 while (workerRunning) {
201 dyad_update();
204 dyad_shutdown();
205 printf("tcpThread end!!\n");
206 return NULL;
209 // system
210 void systemInit(void) {
211 int ret;
213 clock_gettime(CLOCK_MONOTONIC, &start_time);
214 printf("[system]Init...\n");
216 SystemCoreClock = 500 * 1e6; // fake 500MHz
218 if (pthread_mutex_init(&updateLock, NULL) != 0) {
219 printf("Create updateLock error!\n");
220 exit(1);
223 if (pthread_mutex_init(&mainLoopLock, NULL) != 0) {
224 printf("Create mainLoopLock error!\n");
225 exit(1);
228 ret = pthread_create(&tcpWorker, NULL, tcpThread, NULL);
229 if (ret != 0) {
230 printf("Create tcpWorker error!\n");
231 exit(1);
234 ret = udpInit(&pwmLink, "127.0.0.1", 9002, false);
235 printf("init PwmOut UDP link...%d\n", ret);
237 ret = udpInit(&stateLink, NULL, 9003, true);
238 printf("start UDP server...%d\n", ret);
240 ret = pthread_create(&udpWorker, NULL, udpThread, NULL);
241 if (ret != 0) {
242 printf("Create udpWorker error!\n");
243 exit(1);
246 // serial can't been slow down
247 rescheduleTask(TASK_SERIAL, 1);
250 void systemReset(void){
251 printf("[system]Reset!\n");
252 workerRunning = false;
253 pthread_join(tcpWorker, NULL);
254 pthread_join(udpWorker, NULL);
255 exit(0);
257 void systemResetToBootloader(bootloaderRequestType_e requestType) {
258 UNUSED(requestType);
260 printf("[system]ResetToBootloader!\n");
261 workerRunning = false;
262 pthread_join(tcpWorker, NULL);
263 pthread_join(udpWorker, NULL);
264 exit(0);
267 void timerInit(void) {
268 printf("[timer]Init...\n");
271 void timerStart(void) {
274 void failureMode(failureMode_e mode) {
275 printf("[failureMode]!!! %d\n", mode);
276 while (1);
279 void indicateFailure(failureMode_e mode, int repeatCount)
281 UNUSED(repeatCount);
282 printf("Failure LED flash for: [failureMode]!!! %d\n", mode);
285 // Time part
286 // Thanks ArduPilot
287 uint64_t nanos64_real() {
288 struct timespec ts;
289 clock_gettime(CLOCK_MONOTONIC, &ts);
290 return (ts.tv_sec*1e9 + ts.tv_nsec) - (start_time.tv_sec*1e9 + start_time.tv_nsec);
293 uint64_t micros64_real() {
294 struct timespec ts;
295 clock_gettime(CLOCK_MONOTONIC, &ts);
296 return 1.0e6*((ts.tv_sec + (ts.tv_nsec*1.0e-9)) - (start_time.tv_sec + (start_time.tv_nsec*1.0e-9)));
299 uint64_t millis64_real() {
300 struct timespec ts;
301 clock_gettime(CLOCK_MONOTONIC, &ts);
302 return 1.0e3*((ts.tv_sec + (ts.tv_nsec*1.0e-9)) - (start_time.tv_sec + (start_time.tv_nsec*1.0e-9)));
305 uint64_t micros64() {
306 static uint64_t last = 0;
307 static uint64_t out = 0;
308 uint64_t now = nanos64_real();
310 out += (now - last) * simRate;
311 last = now;
313 return out*1e-3;
314 // return micros64_real();
317 uint64_t millis64() {
318 static uint64_t last = 0;
319 static uint64_t out = 0;
320 uint64_t now = nanos64_real();
322 out += (now - last) * simRate;
323 last = now;
325 return out*1e-6;
326 // return millis64_real();
329 uint32_t micros(void) {
330 return micros64() & 0xFFFFFFFF;
333 uint32_t millis(void) {
334 return millis64() & 0xFFFFFFFF;
337 void microsleep(uint32_t usec) {
338 struct timespec ts;
339 ts.tv_sec = 0;
340 ts.tv_nsec = usec*1000UL;
341 while (nanosleep(&ts, &ts) == -1 && errno == EINTR) ;
344 void delayMicroseconds(uint32_t us) {
345 microsleep(us / simRate);
348 void delayMicroseconds_real(uint32_t us) {
349 microsleep(us);
352 void delay(uint32_t ms) {
353 uint64_t start = millis64();
355 while ((millis64() - start) < ms) {
356 microsleep(1000);
360 // Subtract the ‘struct timespec’ values X and Y, storing the result in RESULT.
361 // Return 1 if the difference is negative, otherwise 0.
362 // result = x - y
363 // from: http://www.gnu.org/software/libc/manual/html_node/Elapsed-Time.html
364 int timeval_sub(struct timespec *result, struct timespec *x, struct timespec *y) {
365 unsigned int s_carry = 0;
366 unsigned int ns_carry = 0;
367 // Perform the carry for the later subtraction by updating y.
368 if (x->tv_nsec < y->tv_nsec) {
369 int nsec = (y->tv_nsec - x->tv_nsec) / 1000000000 + 1;
370 ns_carry += 1000000000 * nsec;
371 s_carry += nsec;
374 // Compute the time remaining to wait. tv_usec is certainly positive.
375 result->tv_sec = x->tv_sec - y->tv_sec - s_carry;
376 result->tv_nsec = x->tv_nsec - y->tv_nsec + ns_carry;
378 // Return 1 if result is negative.
379 return x->tv_sec < y->tv_sec;
383 // PWM part
384 pwmOutputPort_t motors[MAX_SUPPORTED_MOTORS];
385 static pwmOutputPort_t servos[MAX_SUPPORTED_SERVOS];
387 // real value to send
388 static int16_t motorsPwm[MAX_SUPPORTED_MOTORS];
389 static int16_t servosPwm[MAX_SUPPORTED_SERVOS];
390 static int16_t idlePulse;
392 void servoDevInit(const servoDevConfig_t *servoConfig) {
393 UNUSED(servoConfig);
394 for (uint8_t servoIndex = 0; servoIndex < MAX_SUPPORTED_SERVOS; servoIndex++) {
395 servos[servoIndex].enabled = true;
399 static motorDevice_t motorPwmDevice; // Forward
401 pwmOutputPort_t *pwmGetMotors(void) {
402 return motors;
405 static float pwmConvertFromExternal(uint16_t externalValue)
407 return (float)externalValue;
410 static uint16_t pwmConvertToExternal(float motorValue)
412 return (uint16_t)motorValue;
415 static void pwmDisableMotors(void)
417 motorPwmDevice.enabled = false;
420 static bool pwmEnableMotors(void)
422 motorPwmDevice.enabled = true;
424 return true;
427 static void pwmWriteMotor(uint8_t index, float value)
429 motorsPwm[index] = value - idlePulse;
432 static void pwmWriteMotorInt(uint8_t index, uint16_t value)
434 pwmWriteMotor(index, (float)value);
437 static void pwmShutdownPulsesForAllMotors(void)
439 motorPwmDevice.enabled = false;
442 bool pwmIsMotorEnabled(uint8_t index) {
443 return motors[index].enabled;
446 static void pwmCompleteMotorUpdate(void)
448 // send to simulator
449 // for gazebo8 ArduCopterPlugin remap, normal range = [0.0, 1.0], 3D rang = [-1.0, 1.0]
451 double outScale = 1000.0;
452 if (featureIsEnabled(FEATURE_3D)) {
453 outScale = 500.0;
456 pwmPkt.motor_speed[3] = motorsPwm[0] / outScale;
457 pwmPkt.motor_speed[0] = motorsPwm[1] / outScale;
458 pwmPkt.motor_speed[1] = motorsPwm[2] / outScale;
459 pwmPkt.motor_speed[2] = motorsPwm[3] / outScale;
461 // get one "fdm_packet" can only send one "servo_packet"!!
462 if (pthread_mutex_trylock(&updateLock) != 0) return;
463 udpSend(&pwmLink, &pwmPkt, sizeof(servo_packet));
464 // printf("[pwm]%u:%u,%u,%u,%u\n", idlePulse, motorsPwm[0], motorsPwm[1], motorsPwm[2], motorsPwm[3]);
467 void pwmWriteServo(uint8_t index, float value) {
468 servosPwm[index] = value;
471 static motorDevice_t motorPwmDevice = {
472 .vTable = {
473 .postInit = motorPostInitNull,
474 .convertExternalToMotor = pwmConvertFromExternal,
475 .convertMotorToExternal = pwmConvertToExternal,
476 .enable = pwmEnableMotors,
477 .disable = pwmDisableMotors,
478 .isMotorEnabled = pwmIsMotorEnabled,
479 .updateStart = motorUpdateStartNull,
480 .write = pwmWriteMotor,
481 .writeInt = pwmWriteMotorInt,
482 .updateComplete = pwmCompleteMotorUpdate,
483 .shutdown = pwmShutdownPulsesForAllMotors,
487 motorDevice_t *motorPwmDevInit(const motorDevConfig_t *motorConfig, uint16_t _idlePulse, uint8_t motorCount, bool useUnsyncedPwm)
489 UNUSED(motorConfig);
490 UNUSED(useUnsyncedPwm);
492 if (motorCount > 4) {
493 return NULL;
496 idlePulse = _idlePulse;
498 for (int motorIndex = 0; motorIndex < MAX_SUPPORTED_MOTORS && motorIndex < motorCount; motorIndex++) {
499 motors[motorIndex].enabled = true;
501 motorPwmDevice.count = motorCount; // Never used, but seemingly a right thing to set it anyways.
502 motorPwmDevice.initialized = true;
503 motorPwmDevice.enabled = false;
505 return &motorPwmDevice;
508 // ADC part
509 uint16_t adcGetChannel(uint8_t channel) {
510 UNUSED(channel);
511 return 0;
514 // stack part
515 char _estack;
516 char _Min_Stack_Size;
518 // fake EEPROM
519 static FILE *eepromFd = NULL;
521 void FLASH_Unlock(void) {
522 if (eepromFd != NULL) {
523 fprintf(stderr, "[FLASH_Unlock] eepromFd != NULL\n");
524 return;
527 // open or create
528 eepromFd = fopen(EEPROM_FILENAME,"r+");
529 if (eepromFd != NULL) {
530 // obtain file size:
531 fseek(eepromFd , 0 , SEEK_END);
532 size_t lSize = ftell(eepromFd);
533 rewind(eepromFd);
535 size_t n = fread(eepromData, 1, sizeof(eepromData), eepromFd);
536 if (n == lSize) {
537 printf("[FLASH_Unlock] loaded '%s', size = %ld / %ld\n", EEPROM_FILENAME, lSize, sizeof(eepromData));
538 } else {
539 fprintf(stderr, "[FLASH_Unlock] failed to load '%s'\n", EEPROM_FILENAME);
540 return;
542 } else {
543 printf("[FLASH_Unlock] created '%s', size = %ld\n", EEPROM_FILENAME, sizeof(eepromData));
544 if ((eepromFd = fopen(EEPROM_FILENAME, "w+")) == NULL) {
545 fprintf(stderr, "[FLASH_Unlock] failed to create '%s'\n", EEPROM_FILENAME);
546 return;
548 if (fwrite(eepromData, sizeof(eepromData), 1, eepromFd) != 1) {
549 fprintf(stderr, "[FLASH_Unlock] write failed: %s\n", strerror(errno));
554 void FLASH_Lock(void) {
555 // flush & close
556 if (eepromFd != NULL) {
557 fseek(eepromFd, 0, SEEK_SET);
558 fwrite(eepromData, 1, sizeof(eepromData), eepromFd);
559 fclose(eepromFd);
560 eepromFd = NULL;
561 printf("[FLASH_Lock] saved '%s'\n", EEPROM_FILENAME);
562 } else {
563 fprintf(stderr, "[FLASH_Lock] eeprom is not unlocked\n");
567 FLASH_Status FLASH_ErasePage(uintptr_t Page_Address) {
568 UNUSED(Page_Address);
569 // printf("[FLASH_ErasePage]%x\n", Page_Address);
570 return FLASH_COMPLETE;
573 FLASH_Status FLASH_ProgramWord(uintptr_t addr, uint32_t value) {
574 if ((addr >= (uintptr_t)eepromData) && (addr < (uintptr_t)ARRAYEND(eepromData))) {
575 *((uint32_t*)addr) = value;
576 printf("[FLASH_ProgramWord]%p = %08x\n", (void*)addr, *((uint32_t*)addr));
577 } else {
578 printf("[FLASH_ProgramWord]%p out of range!\n", (void*)addr);
580 return FLASH_COMPLETE;
583 void IOConfigGPIO(IO_t io, ioConfig_t cfg)
585 UNUSED(io);
586 UNUSED(cfg);
587 printf("IOConfigGPIO\n");
590 void spektrumBind(rxConfig_t *rxConfig)
592 UNUSED(rxConfig);
593 printf("spektrumBind\n");
596 void unusedPinsInit(void)
598 printf("unusedPinsInit\n");