Глобальный массив Arduino не изменяется при доступе через последовательный порт?

Я работаю над системой домашней автоматизации, в состав которой входит Arduino Mega2560, подключенный через USB к RaspberryPi 3 Model B. Arduino получает две короткие, простые команды через серийный номер от Pi; первая команда устанавливает "области" светодиодов, а вторая устанавливает цвета в ранее описанных областях. Синтаксис следующий для обеих команд: элементы в квадратных скобках представляют собой одиночные байты, len+off - 3 байта ascii, интерпретируемые как base-10, а r+g+b - 2 байта ascii, каждый из которых интерпретируется как шестнадцатеричный. Команда color set поддерживает переменное количество областей, как показано:

Регион Set Command

!c[regionId][stripId][len2][len1][len0][off2][off1][off0]\r

Команда цветового набора

!b[r1][r0][g1][g0][b1][b0][numRegions][regionId0]([regionId1]...)\r

Каждая команда отправляет положительные / отрицательные подтверждения, так что вы можете подумать, что я смогу сказать, что происходит, но я получаю неожиданное поведение, когда пытаюсь установить светлые цвета с помощью второй команды, несмотря на получение подтверждений, как и ожидалось. Пока что единственный разумный вывод, к которому я могу прийти, состоит в том, что для областей глобального массива изменения игнорируются и всегда остаются равными нулю. Например, этот обмен приводит к тому, что свет не горит (комментарии не находятся в реальном обмене)

Pi sends: !c00144000\r  // create regionId 0 on stripId 0; length 144 offset 0
Pi recieves: O          // positive confirmation?
Pi sends: !b00009910\r  // set regionId 0 to color 0x000099 (blue)
Pi recieves: O          // positive confirmation, but no lights?
Pi sends: !x\r          // invalid command; triggers default in switch
Pi recieves: X          // as expected, all lights turn red

Вот урезанный источник для Arduino:

#include "Arduino.h"
#include "FastLED.h"
#define WAIT while(!Serial.available()) ;; 

// Forward Declarations
const int max_strips = 2;
const int max_strip_length = 300;
const int max_leds = max_strips * max_strip_length;
CRGB leds[max_strips][max_strip_length];

const int max_regions = 20;
int regions[max_regions][3];

const int baud_rate = 9600;
unsigned char buffer[64];
unsigned char currentByte = 0;
unsigned char incommingByte;
unsigned char startChar = '!';
unsigned char endChar = '\r';
bool store = false;

// Helper Functions
void set(CRGB c) {
  for(int i = 0; i < max_strips; i++) {
      for(int j = 0; j < max_strip_length; j++) {
          leds[i][j] = c;
      }
  }
}

void set(int stripId, int length, int offset, CRGB c) {
    for(int i = offset; i < offset+length; i++) {
        leds[stripId][i] = c;
    }
}

void setup() {
  Serial.begin(baud_rate);
  for(int i = 0; i < max_strips; i++) {
      switch(i) {
          case 0: {
              FastLED.addLeds<WS2812B, 2, GRB>(leds[0], max_strip_length);
          } break;
          case 1: {
              FastLED.addLeds<WS2812B, 3, GRB>(leds[1], max_strip_length);
          } break;
      }
  }
  set(CRGB::Black);
}

void loop() {
    if(Serial.available() > 0) {
        unsigned char incomingByte = Serial.read();
        if(incomingByte == startChar) {
            currentByte = 0;
            store = true;
        }
        if(store) {
            if(incomingByte == endChar) {
                buffer[currentByte++] = incomingByte;
                store = false;
                switch(buffer[1]) {
                    case 'b': {
                        int red = (buffer[2] - '0')*16 + (buffer[3] - '0');
                        int green = (buffer[4] - '0')*16 + (buffer[5] - '0');
                        int blue = (buffer[6] - '0')*16 + (buffer[7] - '0');
                        int numRegions = buffer[8] - '0';

                        for(int i = 0; i < numRegions; i++) {
                            int regionId = buffer[9+i] - '0';

                            if(regionId >= max_regions) {
                                Serial.write('S');
                                return;
                            }

                            // Never sets anything
                            set(regions[regionId][0], regions[regionId][1], regions[regionId][2], CRGB(red, green, blue));
                        }

                        Serial.write('O');
                    } break;

                    case 'c': {
                        int regionId = buffer[2] - '0';
                        int stripId = buffer[3] - '0';
                        int length = (buffer[4] - '0')*100 + (buffer[5] - '0')*10 + (buffer[6] -'0');
                        int offset = (buffer[7] - '0')*100 + (buffer[8] - '0')*10 + (buffer[9] -'0');

                        if(regionId >= max_regions) {
                            Serial.write('R');
                            return;
                        }

                        if(stripId >= max_strips) {
                            Serial.write('S');
                            return;
                        }

                        regions[regionId][0] = stripId; // WE LOSE THESE VALUES??
                        regions[regionId][1] = length;  // ??
                        regions[regionId][2] = offset;  // ??

                        Serial.write('O');

                    } break;

                    default: {
                        set(CRGB::Red);
                        Serial.write('X');
                    } break;
                }
                currentByte = 0;
            }else if(currentByte > 64){
                store = false;
                currentByte = 0;
            } else {
                buffer[currentByte++] = incomingByte;
            }
        }
    }
    FastLED.show();
}

Буду признателен за любые советы по поводу того, что мне здесь не хватает, и в любом случае, спасибо за чтение этого места!

1 ответ

Это произошло из-за того, что у меня появилось много мусора на последовательном интерфейсе для Arduino, на котором я тестирую. Снижение скорости передачи до 2400 бит / с решает проблему на этом устройстве. Другие Arduinos в системе работают хорошо с гораздо более высокими последовательными скоростями - я подозреваю, что аппаратная проблема с кристаллом на этом конкретном Arduino, поэтому тактовые частоты не совпадают правильно между двумя сторонами последовательного соединения.

Другие вопросы по тегам