mirror of
https://github.com/mysensors/MySensors.git
synced 2026-03-05 07:44:14 +01:00
The rules are in .tools/astyle/config/style.cfg and can also be reviewed at https://www.mysensors.org/view/260#coding-guidelines-core-library
229 lines
5.3 KiB
C++
229 lines
5.3 KiB
C++
/**
|
|
* The MySensors Arduino library handles the wireless radio link and protocol
|
|
* between your home built sensors/actuators and HA controller of choice.
|
|
* The sensors forms a self healing radio network with optional repeaters. Each
|
|
* repeater and gateway builds a routing tables in EEPROM which keeps track of the
|
|
* network topology allowing messages to be routed to nodes.
|
|
*
|
|
* Created by Henrik Ekblad <henrik.ekblad@mysensors.org>
|
|
* Copyright (C) 2013-2015 Sensnology AB
|
|
* Full contributor list: https://github.com/mysensors/Arduino/graphs/contributors
|
|
*
|
|
* Documentation: http://www.mysensors.org
|
|
* Support Forum: http://forum.mysensors.org
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License
|
|
* version 2 as published by the Free Software Foundation.
|
|
*/
|
|
|
|
#include "MyHwSAMD.h"
|
|
|
|
|
|
/*
|
|
int8_t pinIntTrigger = 0;
|
|
void wakeUp() //place to send the interrupts
|
|
{
|
|
pinIntTrigger = 1;
|
|
}
|
|
void wakeUp2() //place to send the second interrupts
|
|
{
|
|
pinIntTrigger = 2;
|
|
}
|
|
|
|
// Watchdog Timer interrupt service routine. This routine is required
|
|
// to allow automatic WDIF and WDIE bit clearance in hardware.
|
|
ISR (WDT_vect)
|
|
{
|
|
// WDIE & WDIF is cleared in hardware upon entering this ISR
|
|
wdt_disable();
|
|
}
|
|
*/
|
|
|
|
|
|
|
|
void i2c_eeprom_write_byte(unsigned int eeaddress, byte data )
|
|
{
|
|
int rdata = data;
|
|
Wire.beginTransmission(I2C_EEP_ADDRESS);
|
|
Wire.write((int)(eeaddress >> 8)); // MSB
|
|
Wire.write((int)(eeaddress & 0xFF)); // LSB
|
|
Wire.write(rdata);
|
|
Wire.endTransmission();
|
|
}
|
|
|
|
byte i2c_eeprom_read_byte(unsigned int eeaddress )
|
|
{
|
|
byte rdata = 0xFF;
|
|
Wire.beginTransmission(I2C_EEP_ADDRESS);
|
|
Wire.write((int)(eeaddress >> 8)); // MSB
|
|
Wire.write((int)(eeaddress & 0xFF)); // LSB
|
|
Wire.endTransmission();
|
|
Wire.requestFrom(I2C_EEP_ADDRESS,1);
|
|
if (Wire.available()) {
|
|
rdata = Wire.read();
|
|
}
|
|
return rdata;
|
|
}
|
|
|
|
void hwReadConfigBlock(void* buf, void* adr, size_t length)
|
|
{
|
|
uint8_t* dst = static_cast<uint8_t*>(buf);
|
|
int offs = reinterpret_cast<int>(adr);
|
|
while (length-- > 0) {
|
|
*dst++ = i2c_eeprom_read_byte(offs++);
|
|
}
|
|
}
|
|
|
|
void hwWriteConfigBlock(void* buf, void* adr, size_t length)
|
|
{
|
|
uint8_t* src = static_cast<uint8_t*>(buf);
|
|
int offs = reinterpret_cast<int>(adr);
|
|
while (length-- > 0) {
|
|
i2c_eeprom_write_byte(offs++, *src++);
|
|
}
|
|
}
|
|
|
|
uint8_t hwReadConfig(int adr)
|
|
{
|
|
uint8_t value;
|
|
hwReadConfigBlock(&value, reinterpret_cast<void*>(adr), 1);
|
|
return value;
|
|
}
|
|
|
|
void hwWriteConfig(int adr, uint8_t value)
|
|
{
|
|
uint8_t curr = hwReadConfig(adr);
|
|
if (curr != value) {
|
|
hwWriteConfigBlock(&value, reinterpret_cast<void*>(adr), 1);
|
|
}
|
|
}
|
|
|
|
void hwInit()
|
|
{
|
|
MY_SERIALDEVICE.begin(MY_BAUD_RATE);
|
|
#if defined(MY_GATEWAY_SERIAL)
|
|
while (!MY_SERIALDEVICE) {}
|
|
#endif
|
|
Wire.begin();
|
|
}
|
|
|
|
void hwWatchdogReset()
|
|
{
|
|
// TODO: Not supported!
|
|
}
|
|
|
|
void hwReboot()
|
|
{
|
|
NVIC_SystemReset();
|
|
while (true);
|
|
}
|
|
|
|
int8_t hwSleep(unsigned long ms)
|
|
{
|
|
// TODO: Not supported!
|
|
(void)ms;
|
|
return MY_SLEEP_NOT_POSSIBLE;
|
|
}
|
|
|
|
int8_t hwSleep(uint8_t interrupt, uint8_t mode, unsigned long ms)
|
|
{
|
|
// TODO: Not supported!
|
|
(void)interrupt;
|
|
(void)mode;
|
|
(void)ms;
|
|
return MY_SLEEP_NOT_POSSIBLE;
|
|
}
|
|
|
|
int8_t hwSleep(uint8_t interrupt1, uint8_t mode1, uint8_t interrupt2, uint8_t mode2,
|
|
unsigned long ms)
|
|
{
|
|
// TODO: Not supported!
|
|
(void)interrupt1;
|
|
(void)mode1;
|
|
(void)interrupt2;
|
|
(void)mode2;
|
|
(void)ms;
|
|
return MY_SLEEP_NOT_POSSIBLE;
|
|
}
|
|
|
|
#if defined(MY_DEBUG) || defined(MY_SPECIAL_DEBUG)
|
|
uint16_t hwCPUVoltage()
|
|
{
|
|
|
|
// disable ADC
|
|
while (ADC->STATUS.bit.SYNCBUSY);
|
|
ADC->CTRLA.bit.ENABLE = 0x00;
|
|
|
|
// internal 1V reference (default)
|
|
analogReference(AR_INTERNAL1V0);
|
|
// 12 bit resolution (default)
|
|
analogWriteResolution(12);
|
|
// MUXp 0x1B = SCALEDIOVCC/4 => connected to Vcc
|
|
ADC->INPUTCTRL.bit.MUXPOS = 0x1B ;
|
|
|
|
// enable ADC
|
|
while (ADC->STATUS.bit.SYNCBUSY);
|
|
ADC->CTRLA.bit.ENABLE = 0x01;
|
|
// start conversion
|
|
while (ADC->STATUS.bit.SYNCBUSY);
|
|
ADC->SWTRIG.bit.START = 1;
|
|
// clear the Data Ready flag
|
|
ADC->INTFLAG.bit.RESRDY = 1;
|
|
// start conversion again, since The first conversion after the reference is changed must not be used.
|
|
while (ADC->STATUS.bit.SYNCBUSY);
|
|
ADC->SWTRIG.bit.START = 1;
|
|
|
|
// waiting for conversion to complete
|
|
while (!ADC->INTFLAG.bit.RESRDY);
|
|
const uint32_t valueRead = ADC->RESULT.reg;
|
|
|
|
// disable ADC
|
|
while (ADC->STATUS.bit.SYNCBUSY);
|
|
ADC->CTRLA.bit.ENABLE = 0x00;
|
|
|
|
return valueRead * 4;
|
|
}
|
|
|
|
uint16_t hwCPUFrequency()
|
|
{
|
|
// TODO: currently reporting compile time frequency (in 1/10MHz)
|
|
return F_CPU / 100000UL;
|
|
}
|
|
|
|
uint16_t hwFreeMem()
|
|
{
|
|
// TODO: Not supported!
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
#ifdef MY_DEBUG
|
|
void hwDebugPrint(const char *fmt, ... )
|
|
{
|
|
if (MY_SERIALDEVICE) {
|
|
char fmtBuffer[MY_SERIAL_OUTPUT_SIZE];
|
|
#ifdef MY_GATEWAY_FEATURE
|
|
// prepend debug message to be handled correctly by controller (C_INTERNAL, I_LOG_MESSAGE)
|
|
snprintf(fmtBuffer, sizeof(fmtBuffer), PSTR("0;255;%d;0;%d;"), C_INTERNAL, I_LOG_MESSAGE);
|
|
MY_SERIALDEVICE.print(fmtBuffer);
|
|
#endif
|
|
va_list args;
|
|
va_start (args, fmt );
|
|
#ifdef MY_GATEWAY_FEATURE
|
|
// Truncate message if this is gateway node
|
|
vsnprintf(fmtBuffer, sizeof(fmtBuffer), fmt, args);
|
|
fmtBuffer[sizeof(fmtBuffer) - 2] = '\n';
|
|
fmtBuffer[sizeof(fmtBuffer) - 1] = '\0';
|
|
#else
|
|
vsnprintf(fmtBuffer, sizeof(fmtBuffer), fmt, args);
|
|
#endif
|
|
va_end (args);
|
|
MY_SERIALDEVICE.print(fmtBuffer);
|
|
// MY_SERIALDEVICE.flush();
|
|
|
|
//MY_SERIALDEVICE.write(freeRam());
|
|
}
|
|
}
|
|
#endif
|