Релейная плата делает выходной сигнал слабым

Я использую Arduino Uno и Relay Shield (http://www.seeedstudio.com/blog/2013/02/28/shield-evolution-flexible-add-ons-for-arduino/relay-shield/), а также использую код, похожий на Teapot Demo, в котором светодиод мигает при обнаружении движения. Когда я подключаю shield, светодиод мигает с нормальной яркостью при загрузке кода, но мигает необычайно тускло при индикации движения. Это и внешний, и внутренний светодиодный дисплей для Arduino.

Вот моя текущая настройка:

Я думаю, что это проблема кода, поскольку светодиод явно может нормально мигать и гаснуть. Есть ли какие-нибудь предложения относительно того, что не так или что мне следует сделать?

Ниже представлен мой код:

#include "I2Cdev.h"
#include "MPU6050_6Axis_MotionApps20.h"
#if I2CDEV_IMPLEMENTATION == I2CDEV_ARDUINO_WIRE
    #include "Wire.h"
#endif
MPU6050 mpu;
// раскомментируйте "OUTPUT_READABLE_REALACCEL", если хотите увидеть ускорение
// компоненты с удаленной гравитацией. Эта система отсчета ускорения
// не компенсируется ориентацией, поэтому +X всегда +X согласно
// датчик, только без эффектов гравитации. Если вам нужно ускорение
// с компенсацией ориентации, вместо этого используйте OUTPUT_READABLE_WORLDACCEL.
#define OUTPUT_READABLE_REALACCEL

// раскомментируйте "OUTPUT_TEAPOT", если вы хотите, чтобы вывод соответствовал
// формат, используемый для демонстрации чайника InvenSense
#define OUTPUT_TEAPOT

//-------------------------------------------------------------------------------

const int LED_PIN = 13; // (Arduino — 13, Teensy — 11, Teensy++ — 6)
const int buzzerPin = 12;

bool blinkState = false;

// Переменные управления/состояния MPU
bool dmpReady = false;  // установить true, если инициализация DMP прошла успешно
uint8_t mpuIntStatus;   // содержит фактический байт статуса прерывания от MPU
uint8_t devStatus;      // возвращаем статус после каждой операции устройства (0 = успех, !0 = ошибка)
uint16_t packetSize;    // ожидаемый размер пакета DMP (по умолчанию 42 байта)
uint16_t fifoCount;     // количество всех байтов в данный момент в FIFO
uint8_t fifoBuffer[64]; // буфер хранения FIFO

// переменные ориентации/движения
Quaternion q;           // [w, x, y, z] контейнер кватерниона
VectorInt16 aa;         // [x, y, z] измерения датчика ускорения
VectorInt16 aaReal;     // [x, y, z] измерения датчика ускорения без учета гравитации
VectorInt16 aaWorld;    // [x, y, z] измерения датчика ускорения в мировой системе координат
VectorFloat gravity;    // [x, y, z] вектор гравитации
float euler[3];         // [psi, theta, phi] Контейнер угла Эйлера
float ypr[3];           // [рыскание, тангаж, крен] контейнер рыскания/тангажа/крена и вектор силы тяжести

// структура пакета для демонстрации чайника InvenSense
uint8_t teapotPacket[14] = { '$', 0x02, 0,0, 0,0, 0,0, 0,0, 0x00, 0x00, '\r', '\n' };


// ===========================================================
// === ПРОЦЕДУРА ОБНАРУЖЕНИЯ ПРЕРЫВАНИЯ ===
// ===========================================================

volatile bool mpuInterrupt = false;     // указывает, перешел ли вывод прерывания MPU в высокий уровень
void dmpDataReady() {
    mpuInterrupt = true;
}

// ===========================================================
// === НАЧАЛЬНАЯ НАСТРОЙКА ===
// ============================================================

void setup() {
    // присоединяемся к шине I2C (библиотека I2Cdev не делает этого автоматически)
    #if I2CDEV_IMPLEMENTATION == I2CDEV_ARDUINO_WIRE
        Wire.begin();
        TWBR = 24; // Частота I2C 400 кГц (200 кГц, если процессор 8 МГц)
    #elif I2CDEV_IMPLEMENTATION == I2CDEV_BUILTIN_FASTWIRE
        Fastwire::setup(400, true);
    #endif

    // инициализация последовательной связи
    // (выбрано 115200, так как оно требуется для вывода Teapot Demo, но это
    // на самом деле все зависит от вашего проекта)
    Serial.begin(115200); // это скорость "Бод"
    while (!Serial); // ждем перечисления Leonardo, остальные продолжают немедленно

    // инициализируем устройство
    Serial.println(F("Initializing I2C devices..."));
    mpu.initialize();

    // проверка соединения
    Serial.println(F("Testing device connections..."));
    Serial.println(mpu.testConnection() ? F("MPU6050 connection successful") : F("MPU6050 connection failed"));

    // ждем готовности
    Serial.println(F("\nSend any character to begin DMP programming and demo: "));
    while (Serial.available() && Serial.read()); // пустой буфер
    while (!Serial.available());                 // ждем данные
    while (Serial.available() && Serial.read()); // снова пустой буфер

    // загрузка и настройка DMP
    Serial.println(F("Initializing DMP..."));
    devStatus = mpu.dmpInitialize();

    // укажите здесь свои собственные смещения гироскопа, масштабированные для минимальной чувствительности
    mpu.setXGyroOffset(220);
    mpu.setYGyroOffset(76);
    mpu.setZGyroOffset(-85);
    mpu.setZAccelOffset(1788); // 1688 — заводское значение по умолчанию для моего тестового чипа

    // убеждаемся, что все работает (возвращает 0, если так)
    if (devStatus == 0) {
        // включаем DMP, теперь он готов
        Serial.println(F("Enabling DMP..."));
        mpu.setDMPEnabled(true);

        // включить обнаружение прерываний Arduino
        Serial.println(F("Enabling interrupt detection (Arduino external interrupt 0)..."));
        attachInterrupt(0, dmpDataReady, RISING);
        mpuIntStatus = mpu.getIntStatus();

        // устанавливаем наш флаг готовности DMP, чтобы основная функция loop() знала, что ее можно использовать
        Serial.println(F("DMP ready! Waiting for first interrupt..."));
        dmpReady = true;

        // получить ожидаемый размер пакета DMP для последующего сравнения
        packetSize = mpu.dmpGetFIFOPacketSize();
    } else {
        // ОШИБКА!
        // 1 = начальная загрузка памяти не удалась
        // 2 = Не удалось обновить конфигурацию DMP
        // (если он сломается, обычно код будет 1)
        Serial.print(F("DMP Initialization failed (code "));
        Serial.print(devStatus);
        Serial.println(F(")"));
    }

    // настроить светодиод для выхода
    pinMode(LED_PIN, INPUT);
    pinMode(buzzerPin, INPUT);
}

// ===========================================================
// === ОСНОВНОЙ ЦИКЛ ПРОГРАММЫ ===
// ============================================================

void loop() {
    // если программирование не удалось, не пытайтесь ничего делать
    if (!dmpReady) return;

    // ждем прерывания MPU или появления дополнительных пакетов
    while (!mpuInterrupt && fifoCount < packetSize) {
        // здесь есть другие сведения о поведении программы
    }

    // сбрасываем флаг прерывания и получаем байт INT_STATUS
    mpuInterrupt = false;
    mpuIntStatus = mpu.getIntStatus();

    // получить текущий счетчик FIFO
    fifoCount = mpu.getFIFOCount();

    // проверка на переполнение (это никогда не должно происходить, если только наш код не слишком неэффективен)
    if ((mpuIntStatus & 0x10) || fifoCount == 1024) {
        // сброс, чтобы мы могли продолжить работу без ошибок
        mpu.resetFIFO();
        Serial.println(F("FIFO overflow!"));

    // в противном случае проверьте прерывание готовности данных DMP (это должно происходить часто)
    } else if (mpuIntStatus & 0x02) {
        // ждем корректной длины доступных данных, ожидание должно быть ОЧЕНЬ коротким
        while (fifoCount < packetSize) fifoCount = mpu.getFIFOCount();

        // прочитать пакет из FIFO
        mpu.getFIFOBytes(fifoBuffer, packetSize);

        // отслеживайте количество FIFO здесь на случай, если доступен > 1 пакет
        // (это позволяет нам немедленно прочитать больше, не дожидаясь прерывания)
        fifoCount -= packetSize;

        #ifdef OUTPUT_READABLE_REALACCEL
            // отображать реальное ускорение, скорректированное для устранения гравитации
            int Mag;
            mpu.dmpGetQuaternion(&q, fifoBuffer);
            mpu.dmpGetAccel(&aa, fifoBuffer);
            mpu.dmpGetGravity(&gravity, &q);
            mpu.dmpGetLinearAccel(&aaReal, &aa, &gravity);
            Serial.print("areal\t");
            Serial.print(aaReal.x);
            Serial.print("\t");
            Serial.print(aaReal.y);
            Serial.print("\n");
            Mag = sqrt((aaReal.x*aaReal.x) + (aaReal.y*aaReal.y));
            Serial.print("Accel. Mag. is ");
            Serial.print(Mag);
// Serial.print("\t");
// Serial.println(aaReal.z);
        #endif

        #ifdef OUTPUT_TEAPOT
            // отображать значения кватерниона в демонстрационном формате InvenSense Teapot:
            teapotPacket[2] = fifoBuffer[0];
            teapotPacket[3] = fifoBuffer[1];
            teapotPacket[4] = fifoBuffer[4];
            teapotPacket[5] = fifoBuffer[5];
            teapotPacket[6] = fifoBuffer[8];
            teapotPacket[7] = fifoBuffer[9];
            teapotPacket[8] = fifoBuffer[12];
            teapotPacket[9] = fifoBuffer[13];
            Serial.write(teapotPacket, 14);
            teapotPacket[11]++; // packetCount, зацикливается на 0xFF намеренно
        #endif

        // мигание светодиода и звуковой сигнал для индикации активности
        if (Mag > 120) {
          blinkState = !blinkState;
          digitalWrite(LED_PIN, blinkState);
// задержка(1000);
          tone(buzzerPin, 100);
        }
        else if (Mag <= 120) {
          digitalWrite(LED_PIN, LOW);
          noTone(buzzerPin);
        }

    }
}

, 👍1


1 ответ


Лучший ответ:

1

Не видя вашего кода, сложно сказать вам напрямую, в чем проблема. Если можете, пожалуйста, опубликуйте свой код.

Основываясь на информации, изложенной в вашем вопросе, я вижу две возможные причины этого.

Во-первых, может быть, что мощности вашего источника питания недостаточно для питания всего. Поэтому, когда Arduino пытается включить реле, у него не хватает мощности, напряжение падает слишком низко, и Arduino сбрасывается, обесточивая реле. Это происходит много раз в секунду и отображается тусклым светодиодом.

Это можно проверить, отключив реле от выхода Arduino. Запустите скетч еще раз, и если все работает на полной яркости без подключенного реле, то, скорее всего, мощности недостаточно.

Моя вторая теория заключается в том, что это ошибка в вашем коде. В некоторых циклах он обнаруживает движение и поэтому включает светодиод. В других циклах он не обнаруживает изменение движения и поэтому выключает светодиод. Это происходит много тысяч раз в секунду. Это было бы видно на осциллографе, если бы он у вас был.

Простейшее, что вы могли бы сделать для этого сценария, — это добавить задержку (например, delay(1000);) сразу после включения светодиода при обнаружении движения. Это должно оставить светодиод включенным по крайней мере на секунду после обнаружения движения .

,